Querying: Sorting
Basics
Starting from RavenDB 4.0, the server will determine possible sorting capabilities automatically from the indexed value, but sorting will not be applied until you request it by using the appropriate methods. The following queries will not return ordered results:
const results = await session
.query({ indexName: "Products/ByUnitsInStock" })
.whereGreaterThan("UnitsInStock", 10)
.all();
class Products_ByUnitsInStock extends AbstractIndexCreationTask {
constructor() {
super();
this.map = `docs.Products.Select(product => new {
UnitsInStock = product.UnitsInStock
})`;
}
}
from index 'Products/ByUnitsInStock'
where UnitsInStock > 10
To start sorting, we need to request to order by some specified index field. In our case we will order by UnitsInStock
in descending order:
const results = await session
.query({ indexName: "Products/ByUnitsInStock" })
.whereGreaterThan("UnitsInStock", 10)
.orderByDescending("UnitsInStock")
.all();
class Products_ByUnitsInStock extends AbstractIndexCreationTask {
constructor() {
super();
this.map = `docs.Products.Select(product => new {
UnitsInStock = product.UnitsInStock
})`;
}
}
from index 'Products/ByUnitsInStock'
where UnitsInStock > 10
order by UnitsInStock as long desc
Forcing ordering type
OrderingType
can be forced by passing OrderingType
explicitly to one of the orderBy()
methods.
const results = await session
.query({ indexName: "Products/ByUnitsInStock" })
.whereGreaterThan("UnitsInStock", 10)
.orderByDescending("UnitsInStock", "String")
.all();
class Products_ByUnitsInStock extends AbstractIndexCreationTask {
constructor() {
super();
this.map = `docs.Products.Select(product => new {
UnitsInStock = product.UnitsInStock
})`;
}
}
from index 'Products/ByUnitsInStock'
where UnitsInStock > 10
order by UnitsInStock desc
Ordering by Score
When a query is issued, each index entry is scored by Lucene (you can read more about Lucene scoring here) and this value is available in metadata information of a document under @index-score
(the higher the value, the better the match). To order by this value you can use the orderByScore()
or the orderByScoreDescending()
methods:
const results = await session
.query({ indexName: "Products/ByUnitsInStock" })
.whereGreaterThan("UnitsInStock", 10)
.orderByScore()
.all();
class Products_ByUnitsInStock extends AbstractIndexCreationTask {
constructor() {
super();
this.map = `docs.Products.Select(product => new {
UnitsInStock = product.UnitsInStock
})`;
}
}
from index 'Products/ByUnitsInStock'
where UnitsInStock > 10
order by score()
Chaining Orderings
It is also possible to chain multiple orderings of the query results.
You can sort the query results first by some specified index field (or by the @index-score
), then sort all the equal entries by some different index field (or the @index-score
).
This can be achieved by using the orderBy()
(orderByDescending()
) and orderByScore()
(orderByScoreDescending()
) methods.
const results = await session
.query({ indexName: "Products/ByUnitsInStockAndName" })
.whereGreaterThan("UnitsInStock", 10)
.orderBy("UnitsInStock")
.orderByScore()
.orderByDescending("Name")
.all();
class Products_ByUnitsInStockAndName extends AbstractIndexCreationTask {
constructor() {
super();
this.map = `docs.Products.Select(product => new {
UnitsInStock = product.UnitsInStock,
Name = product.Name
})`;
}
}
from index 'Products/ByUnitsInStockAndName'
where UnitsInStock > 10
order by UnitsInStock, score(), Name desc
Random Ordering
If you want to randomize the order of your results each time the query is executed, use the randomOrdering()
method (API reference here):
const results = await session
.query({ indexName: "Products/ByUnitsInStock" })
.randomOrdering()
.whereGreaterThan("UnitsInStock", 10)
.all();
class Products_ByUnitsInStock extends AbstractIndexCreationTask {
constructor() {
super();
this.map = `docs.Products.Select(product => new {
UnitsInStock = product.UnitsInStock
})`;
}
}
from index 'Products/ByUnitsInStock'
where UnitsInStock > 10
order by random()
Ordering When a Field is Searchable
When sorting must be done on field that is Searchable, due to Lucene limitations sorting on such a field is not supported. To overcome this, create another field that is not searchable, and sort by it.
const results = await session
.query({ indexName: "Products/ByName/Search" })
.search("Name", "Louisiana")
.orderByDescending("NameForSorting")
.all();
class Products_ByName_Search extends AbstractIndexCreationTask {
constructor() {
super();
this.map = `docs.Products.Select(product => new {
Name = product.Name,
NameForSorting = product.Name
})`;
this.index("Name", "Search");
}
}
from index 'Products/ByName/Search'
where search(Name, 'Louisiana')
order by NameForSorting desc
AlphaNumeric Ordering
Sometimes when ordering strings, it doesn't make sense to use the default lexicographic ordering.
For example, "Abc9" will come after "Abc10" because if treated as single characters, 9 is greater than 1.
If you want digit characters in a string to be treated as numbers and not as text, you should use alphanumeric ordering. In that case, when comparing "Abc10" to "Abc9", the digits 1 and 0 will be treated as the number 10 which will be considered greater than 9.
To order in this mode you can pass the "AlphaNumeric"
type into orderBy()
or orderByDescending()
:
const results = await session
.query({ indexName: "Products/ByUnitsInStock" })
.whereGreaterThan("UnitsInStock", 10)
.orderBy("Name", "AlphaNumeric")
.all();
class Products_ByUnitsInStock extends AbstractIndexCreationTask {
constructor() {
super();
this.map = `docs.Products.Select(product => new {
UnitsInStock = product.UnitsInStock
})`;
}
}
from index 'Products/ByUnitsInStock '
where UnitsInStock > 10
order by Name as alphanumeric