Skip to main content

sea_orm/
lib.rs

1#![cfg_attr(docsrs, feature(doc_cfg))]
2#![warn(missing_docs)]
3#![deny(
4    missing_debug_implementations,
5    clippy::missing_panics_doc,
6    clippy::unwrap_used,
7    clippy::print_stderr,
8    clippy::print_stdout
9)]
10
11//! <div align="center">
12//!
13//!   <img alt="SeaORM" src="https://www.sea-ql.org/blog/img/SeaORM 2.0 Banner.png"/>
14//!
15//!   <h1></h1>
16//!   <h3>SeaORM is a powerful ORM for building web services in Rust</h3>
17//!
18//!   [![crate](https://img.shields.io/crates/v/sea-orm.svg)](https://crates.io/crates/sea-orm)
19//!   [![build status](https://github.com/SeaQL/sea-orm/actions/workflows/rust.yml/badge.svg)](https://github.com/SeaQL/sea-orm/actions/workflows/rust.yml)
20//!   [![GitHub stars](https://img.shields.io/github/stars/SeaQL/sea-orm.svg?style=social&label=Star&maxAge=1)](https://github.com/SeaQL/sea-orm/stargazers/)
21//!   <br>Support us with a ⭐ !
22//!
23//! </div>
24//!
25//! # 🐚 SeaORM
26//!
27//! [δΈ­ζ–‡ζ–‡ζ‘£](https://github.com/SeaQL/sea-orm/blob/master/README-zh.md)
28//!
29//! ### Advanced Relations
30//!
31//! Model complex relationships 1-1, 1-N, M-N, and even self-referential in a high-level, conceptual way.
32//!
33//! ### Familiar Concepts
34//!
35//! Inspired by popular ORMs in the Ruby, Python, and Node.js ecosystem, SeaORM offers a developer experience that feels instantly recognizable.
36//!
37//! ### Feature Rich
38//!
39//! SeaORM is a batteries-included ORM with filters, pagination, and nested queries to accelerate building REST, GraphQL, and gRPC APIs.
40//!
41//! ### Production Ready
42//!
43//! With 250k+ weekly downloads, SeaORM is production-ready, trusted by startups and enterprises worldwide.
44//!
45//! ## Getting Started
46//!
47//! [![Discord](https://img.shields.io/discord/873880840487206962?label=Discord)](https://discord.com/invite/uCPdDXzbdv)
48//! Join our Discord server to chat with others!
49//!
50//! + [Documentation](https://www.sea-ql.org/SeaORM)
51//!
52//! Integration examples:
53//!
54//! + [Actix Example](https://github.com/SeaQL/sea-orm/tree/master/examples/actix_example)
55//! + [Axum Example](https://github.com/SeaQL/sea-orm/tree/master/examples/axum_example)
56//! + [GraphQL Example](https://github.com/SeaQL/sea-orm/tree/master/examples/graphql_example)
57//! + [jsonrpsee Example](https://github.com/SeaQL/sea-orm/tree/master/examples/jsonrpsee_example)
58//! + [Loco Example](https://github.com/SeaQL/sea-orm/tree/master/examples/loco_example) / [Loco REST Starter](https://github.com/SeaQL/sea-orm/tree/master/examples/loco_starter)
59//! + [Poem Example](https://github.com/SeaQL/sea-orm/tree/master/examples/poem_example)
60//! + [Rocket Example](https://github.com/SeaQL/sea-orm/tree/master/examples/rocket_example) / [Rocket OpenAPI Example](https://github.com/SeaQL/sea-orm/tree/master/examples/rocket_okapi_example)
61//! + [Salvo Example](https://github.com/SeaQL/sea-orm/tree/master/examples/salvo_example)
62//! + [Tonic Example](https://github.com/SeaQL/sea-orm/tree/master/examples/tonic_example)
63//! + [Seaography Example (Bakery)](https://github.com/SeaQL/sea-orm/tree/master/examples/seaography_example) / [Seaography Example (Sakila)](https://github.com/SeaQL/seaography/tree/main/examples/sqlite)
64//!
65//! If you want a simple, clean example that fits in a single file that demonstrates the best of SeaORM, you can try:
66//! + [Quickstart](https://github.com/SeaQL/sea-orm/blob/master/examples/quickstart/src/main.rs)
67//!
68//! Let's have a quick walk through of the unique features of SeaORM.
69//!
70//! ## Expressive Entity format
71//! You don't have to write this by hand! Entity files can be generated from an existing database using `sea-orm-cli`,
72//! following is generated with `--entity-format dense` *(new in 2.0)*.
73//! ```
74//! # #[cfg(feature = "macros")]
75//! # mod entities {
76//! # mod profile {
77//! # use sea_orm::entity::prelude::*;
78//! # #[sea_orm::model]
79//! # #[derive(Clone, Debug, PartialEq, Eq, DeriveEntityModel)]
80//! # #[sea_orm(table_name = "profile")]
81//! # pub struct Model {
82//! #     #[sea_orm(primary_key)]
83//! #     pub id: i32,
84//! #     pub picture: String,
85//! #     #[sea_orm(unique)]
86//! #     pub user_id: i32,
87//! #     #[sea_orm(belongs_to, from = "user_id", to = "id")]
88//! #     pub user: HasOne<super::user::Entity>,
89//! # }
90//! # impl ActiveModelBehavior for ActiveModel {}
91//! # }
92//! # mod tag {
93//! # use sea_orm::entity::prelude::*;
94//! # #[sea_orm::model]
95//! # #[derive(Clone, Debug, PartialEq, Eq, DeriveEntityModel)]
96//! # #[sea_orm(table_name = "post")]
97//! # pub struct Model {
98//! #     #[sea_orm(primary_key)]
99//! #     pub id: i32,
100//! #     #[sea_orm(has_many, via = "post_tag")]
101//! #     pub tags: HasMany<super::tag::Entity>,
102//! # }
103//! # impl ActiveModelBehavior for ActiveModel {}
104//! # }
105//! # mod post_tag {
106//! # use sea_orm::entity::prelude::*;
107//! # #[sea_orm::model]
108//! # #[derive(Clone, Debug, PartialEq, DeriveEntityModel, Eq)]
109//! # #[sea_orm(table_name = "post_tag")]
110//! # pub struct Model {
111//! #     #[sea_orm(primary_key, auto_increment = false)]
112//! #     pub post_id: i32,
113//! #     #[sea_orm(primary_key, auto_increment = false)]
114//! #     pub tag_id: i32,
115//! #     #[sea_orm(belongs_to, from = "post_id", to = "id")]
116//! #     pub post: Option<super::post::Entity>,
117//! #     #[sea_orm(belongs_to, from = "tag_id", to = "id")]
118//! #     pub tag: Option<super::tag::Entity>,
119//! # }
120//! # impl ActiveModelBehavior for ActiveModel {}
121//! # }
122//! mod user {
123//!     use sea_orm::entity::prelude::*;
124//!
125//!     #[sea_orm::model]
126//!     #[derive(Clone, Debug, PartialEq, Eq, DeriveEntityModel)]
127//!     #[sea_orm(table_name = "user")]
128//!     pub struct Model {
129//!         #[sea_orm(primary_key)]
130//!         pub id: i32,
131//!         pub name: String,
132//!         #[sea_orm(unique)]
133//!         pub email: String,
134//!         #[sea_orm(has_one)]
135//!         pub profile: HasOne<super::profile::Entity>,
136//!         #[sea_orm(has_many)]
137//!         pub posts: HasMany<super::post::Entity>,
138//!     }
139//! # impl ActiveModelBehavior for ActiveModel {}
140//! }
141//! mod post {
142//!     use sea_orm::entity::prelude::*;
143//!
144//!     #[sea_orm::model]
145//!     #[derive(Clone, Debug, PartialEq, Eq, DeriveEntityModel)]
146//!     #[sea_orm(table_name = "post")]
147//!     pub struct Model {
148//!         #[sea_orm(primary_key)]
149//!         pub id: i32,
150//!         pub user_id: i32,
151//!         pub title: String,
152//!         #[sea_orm(belongs_to, from = "user_id", to = "id")]
153//!         pub author: HasOne<super::user::Entity>,
154//!         #[sea_orm(has_many, via = "post_tag")] // M-N relation with junction
155//!         pub tags: HasMany<super::tag::Entity>,
156//!     }
157//! # impl ActiveModelBehavior for ActiveModel {}
158//! }
159//! # }
160//! ```
161//!
162//! ## Smart Entity Loader
163//! The Entity Loader intelligently uses join for 1-1 and data loader for 1-N relations,
164//! eliminating the N+1 problem even when performing nested queries.
165//! ```
166//! # use sea_orm::{DbConn, DbErr, prelude::*, entity::*, query::*, tests_cfg::*};
167//! # fn function(db: &DbConn) -> Result<(), DbErr> {
168//! // join paths:
169//! // user -> profile
170//! // user -> post
171//! //         post -> post_tag -> tag
172//! let smart_user = user::Entity::load()
173//!     .filter_by_id(42) // shorthand for .filter(user::COLUMN.id.eq(42))
174//!     .with(profile::Entity) // 1-1 uses join
175//!     .with((post::Entity, tag::Entity)) // 1-N uses data loader
176//!     .one(db)?
177//!     .unwrap();
178//!
179//! // 3 queries are executed under the hood:
180//! // 1. SELECT FROM user JOIN profile WHERE id = $
181//! // 2. SELECT FROM post WHERE user_id IN (..)
182//! // 3. SELECT FROM tag JOIN post_tag WHERE post_id IN (..)
183//!
184//! smart_user
185//!     == user::ModelEx {
186//!         id: 42,
187//!         name: "Bob".into(),
188//!         email: "bob@sea-ql.org".into(),
189//!         profile: HasOne::Loaded(
190//!             profile::ModelEx {
191//! #           id: 1,
192//!                 picture: "image.jpg".into(),
193//! #           user_id: 1,
194//! #           user: HasOne::Unloaded,
195//!             }
196//!             .into(),
197//!         ),
198//!         posts: HasMany::Loaded(vec![post::ModelEx {
199//! #           id: 2,
200//! #           user_id: 1,
201//!             title: "Nice weather".into(),
202//! #           author: HasOne::Unloaded,
203//! #           comments: HasMany::Unloaded,
204//!             tags: HasMany::Loaded(vec![tag::ModelEx {
205//! #               id: 3,
206//!                 tag: "sunny".into(),
207//! #               posts: HasMany::Unloaded,
208//!             }]),
209//!         }]),
210//!     };
211//! # Ok(())
212//! # }
213//! ```
214//!
215//! ## ActiveModel: nested persistence made simple
216//! Persist an entire object graph: user, profile (1-1), posts (1-N), and tags (M-N)
217//! in a single operation using a fluent builder API. SeaORM automatically determines
218//! the dependencies and inserts or deletes objects in the correct order.
219//!
220//! ```
221//! # use sea_orm::{DbConn, DbErr, entity::*, query::*, tests_cfg::*};
222//! # fn function(db: &DbConn) -> Result<(), DbErr> {
223//! // this creates the nested object as shown above:
224//! let user = user::ActiveModel::builder()
225//!     .set_name("Bob")
226//!     .set_email("bob@sea-ql.org")
227//!     .set_profile(profile::ActiveModel::builder().set_picture("image.jpg"))
228//!     .add_post(
229//!         post::ActiveModel::builder()
230//!             .set_title("Nice weather")
231//!             .add_tag(tag::ActiveModel::builder().set_tag("sunny")),
232//!     )
233//!     .save(db)?;
234//! # Ok(())
235//! # }
236//! ```
237//!
238//! ## Schema first or Entity first? Your choice
239//!
240//! SeaORM provides a powerful migration system that lets you create tables, modify schemas, and seed data with ease.
241//!
242//! With SeaORM 2.0, you also get a first-class [Entity First Workflow](https://www.sea-ql.org/blog/2025-10-30-sea-orm-2.0/):
243//! simply define new entities or add columns to existing ones,
244//! and SeaORM will automatically detect the changes and create the new tables, columns, unique keys, and foreign keys.
245//!
246//! ```ignore
247//! // SeaORM resolves foreign key dependencies and creates the tables in topological order.
248//! // Requires the `entity-registry` and `schema-sync` feature flags.
249//! db.get_schema_registry("my_crate::entity::*").sync(db);
250//! ```
251//!
252//! ## Ergonomic Raw SQL
253//!
254//! Let SeaORM handle 95% of your transactional queries.
255//! For the remaining cases that are too complex to express,
256//! SeaORM still offers convenient support for writing raw SQL.
257//! ```
258//! # use sea_orm::{DbErr, DbConn};
259//! # fn function(db: &DbConn) -> Result<(), DbErr> {
260//! # use sea_orm::{entity::*, query::*, tests_cfg::*, raw_sql};
261//! # struct Item<'a> { name: &'a str }
262//! let user = Item { name: "Bob" }; // nested parameter access
263//! let ids = [2, 3, 4]; // expanded by the `..` operator
264//!
265//! let user: Option<user::Model> = user::Entity::find()
266//!     .from_raw_sql(raw_sql!(
267//!         Sqlite,
268//!         r#"SELECT "id", "name" FROM "user"
269//!            WHERE "name" LIKE {user.name}
270//!            AND "id" in ({..ids})
271//!         "#
272//!     ))
273//!     .one(db)?;
274//! # Ok(())
275//! # }
276//! ```
277//!
278//! ## Synchronous Support
279//!
280//! [`sea-orm-sync`](https://crates.io/crates/sea-orm-sync) provides the full SeaORM API without requiring an runtime, making it ideal for lightweight CLI programs with SQLite.
281//!
282//! See the [quickstart example](https://github.com/SeaQL/sea-orm/blob/master/sea-orm-sync/examples/quickstart/src/main.rs) for usage.
283//!
284//! ## Basics
285//!
286//! ### Select
287//! SeaORM models 1-N and M-N relationships at the Entity level,
288//! letting you traverse many-to-many links through a junction table in a single call.
289//! ```
290//! # use sea_orm::{DbConn, DbErr, entity::*, query::*, tests_cfg::*};
291//! # fn function(db: &DbConn) -> Result<(), DbErr> {
292//! // find all models
293//! let cakes: Vec<cake::Model> = Cake::find().all(db)?;
294//!
295//! // find and filter
296//! let chocolate: Vec<cake::Model> = Cake::find()
297//!     .filter(Cake::COLUMN.name.contains("chocolate"))
298//!     .all(db)?;
299//!
300//! // find one model
301//! let cheese: Option<cake::Model> = Cake::find_by_id(1).one(db)?;
302//! let cheese: cake::Model = cheese.unwrap();
303//!
304//! // find related models (lazy)
305//! let fruit: Option<fruit::Model> = cheese.find_related(Fruit).one(db)?;
306//!
307//! // find related models (eager): for 1-1 relations
308//! let cake_with_fruit: Vec<(cake::Model, Option<fruit::Model>)> =
309//!     Cake::find().find_also_related(Fruit).all(db)?;
310//!
311//! // find related models (eager): works for both 1-N and M-N relations
312//! let cake_with_fillings: Vec<(cake::Model, Vec<filling::Model>)> = Cake::find()
313//!     .find_with_related(Filling) // for M-N relations, two joins are performed
314//!     .all(db) // rows are automatically consolidated by left entity
315//!     ?;
316//! # Ok(())
317//! # }
318//! ```
319//! ### Nested Select
320//!
321//! Partial models prevent overfetching by letting you querying only the fields
322//! you need; it also makes writing deeply nested relational queries simple.
323//! ```
324//! # use sea_orm::{DbConn, DbErr, entity::*, query::*, tests_cfg::*};
325//! # fn function(db: &DbConn) -> Result<(), DbErr> {
326//! use sea_orm::DerivePartialModel;
327//!
328//! #[derive(DerivePartialModel)]
329//! #[sea_orm(entity = "cake::Entity")]
330//! struct CakeWithFruit {
331//!     id: i32,
332//!     name: String,
333//!     #[sea_orm(nested)]
334//!     fruit: Option<fruit::Model>, // this can be a regular or another partial model
335//! }
336//!
337//! let cakes: Vec<CakeWithFruit> = Cake::find()
338//!     .left_join(fruit::Entity) // no need to specify join condition
339//!     .into_partial_model() // only the columns in the partial model will be selected
340//!     .all(db)?;
341//! # Ok(())
342//! # }
343//! ```
344//!
345//! ### Insert
346//! SeaORM's ActiveModel lets you work directly with Rust data structures and
347//! persist them through a simple API.
348//! It's easy to insert large batches of rows from different data sources.
349//! ```
350//! # use sea_orm::{DbConn, DbErr, entity::*, query::*, tests_cfg::*};
351//! # fn function(db: &DbConn) -> Result<(), DbErr> {
352//! let apple = fruit::ActiveModel {
353//!     name: Set("Apple".to_owned()),
354//!     ..Default::default() // no need to set primary key
355//! };
356//!
357//! let pear = fruit::ActiveModel {
358//!     name: Set("Pear".to_owned()),
359//!     ..Default::default()
360//! };
361//!
362//! // insert one: Active Record style
363//! let apple = apple.insert(db)?;
364//! apple.id == 1;
365//! # let apple = fruit::ActiveModel {
366//! #     name: Set("Apple".to_owned()),
367//! #     ..Default::default() // no need to set primary key
368//! # };
369//!
370//! // insert one: repository style
371//! let result = Fruit::insert(apple).exec(db)?;
372//! result.last_insert_id == 1;
373//! # let apple = fruit::ActiveModel {
374//! #     name: Set("Apple".to_owned()),
375//! #     ..Default::default() // no need to set primary key
376//! # };
377//!
378//! // insert many returning last insert id
379//! let result = Fruit::insert_many([apple, pear]).exec(db)?;
380//! result.last_insert_id == Some(2);
381//! # Ok(())
382//! # }
383//! ```
384//!
385//! ### Insert (advanced)
386//! You can take advantage of database specific features to perform upsert and idempotent insert.
387//! ```
388//! # use sea_orm::{DbConn, TryInsertResult, DbErr, entity::*, query::*, tests_cfg::*};
389//! # fn function_1(db: &DbConn) -> Result<(), DbErr> {
390//! # let apple = fruit::ActiveModel {
391//! #     name: Set("Apple".to_owned()),
392//! #     ..Default::default() // no need to set primary key
393//! # };
394//! # let pear = fruit::ActiveModel {
395//! #     name: Set("Pear".to_owned()),
396//! #     ..Default::default()
397//! # };
398//! // insert many with returning (if supported by database)
399//! let models: Vec<fruit::Model> = Fruit::insert_many([apple, pear]).exec_with_returning(db)?;
400//! models[0]
401//!     == fruit::Model {
402//!         id: 1, // database assigned value
403//!         name: "Apple".to_owned(),
404//!         cake_id: None,
405//!     };
406//! # Ok(())
407//! # }
408//!
409//! # fn function_2(db: &DbConn) -> Result<(), DbErr> {
410//! # let apple = fruit::ActiveModel {
411//! #     name: Set("Apple".to_owned()),
412//! #     ..Default::default() // no need to set primary key
413//! # };
414//! # let pear = fruit::ActiveModel {
415//! #     name: Set("Pear".to_owned()),
416//! #     ..Default::default()
417//! # };
418//! // insert with ON CONFLICT on primary key do nothing, with MySQL specific polyfill
419//! let result = Fruit::insert_many([apple, pear])
420//!     .on_conflict_do_nothing()
421//!     .exec(db)?;
422//!
423//! matches!(result, TryInsertResult::Conflicted);
424//! # Ok(())
425//! # }
426//! ```
427//!
428//! ### Update
429//! ActiveModel avoids race conditions by updating only the fields you've changed,
430//! never overwriting untouched columns.
431//! You can also craft complex bulk update queries with a fluent query building API.
432//! ```
433//! # use sea_orm::{DbConn, DbErr, entity::*, query::*, tests_cfg::*};
434//! use sea_orm::sea_query::{Expr, Value};
435//!
436//! # fn function(db: &DbConn) -> Result<(), DbErr> {
437//! let pear: Option<fruit::Model> = Fruit::find_by_id(1).one(db)?;
438//! let mut pear: fruit::ActiveModel = pear.unwrap().into();
439//!
440//! pear.name = Set("Sweet pear".to_owned()); // update value of a single field
441//!
442//! // update one: only changed columns will be updated
443//! let pear: fruit::Model = pear.update(db)?;
444//!
445//! // update many: UPDATE "fruit" SET "cake_id" = "cake_id" + 2
446//! //               WHERE "fruit"."name" LIKE '%Apple%'
447//! Fruit::update_many()
448//!     .col_expr(fruit::COLUMN.cake_id, fruit::COLUMN.cake_id.add(2))
449//!     .filter(fruit::COLUMN.name.contains("Apple"))
450//!     .exec(db)?;
451//! # Ok(())
452//! # }
453//! ```
454//! ### Save
455//! You can perform "insert or update" operation with ActiveModel, making it easy to compose transactional operations.
456//! ```
457//! # use sea_orm::{DbConn, DbErr, entity::*, query::*, tests_cfg::*};
458//! # fn function(db: &DbConn) -> Result<(), DbErr> {
459//! let banana = fruit::ActiveModel {
460//!     id: NotSet,
461//!     name: Set("Banana".to_owned()),
462//!     ..Default::default()
463//! };
464//!
465//! // create, because primary key `id` is `NotSet`
466//! let mut banana = banana.save(db)?;
467//!
468//! banana.id == Unchanged(2);
469//! banana.name = Set("Banana Mongo".to_owned());
470//!
471//! // update, because primary key `id` is present
472//! let banana = banana.save(db)?;
473//! # Ok(())
474//! # }
475//! ```
476//! ### Delete
477//! The same ActiveModel API consistent with insert and update.
478//! ```
479//! # use sea_orm::{DbConn, DbErr, entity::*, query::*, tests_cfg::*};
480//! # fn function(db: &DbConn) -> Result<(), DbErr> {
481//! // delete one: Active Record style
482//! let orange: Option<fruit::Model> = Fruit::find_by_id(1).one(db)?;
483//! let orange: fruit::Model = orange.unwrap();
484//! orange.delete(db)?;
485//!
486//! // delete one: repository style
487//! let orange = fruit::ActiveModel {
488//!     id: Set(2),
489//!     ..Default::default()
490//! };
491//! fruit::Entity::delete(orange).exec(db)?;
492//!
493//! // delete many: DELETE FROM "fruit" WHERE "fruit"."name" LIKE '%Orange%'
494//! fruit::Entity::delete_many()
495//!     .filter(fruit::COLUMN.name.contains("Orange"))
496//!     .exec(db)?;
497//!
498//! # Ok(())
499//! # }
500//! ```
501//! ### Raw SQL Query
502//! The `raw_sql!` macro is like the `format!` macro but without the risk of SQL injection.
503//! It supports nested parameter interpolation, array and tuple expansion, and even repeating group,
504//! offering great flexibility in crafting complex queries.
505//!
506//! ```
507//! # use sea_orm::{DbErr, DbConn};
508//! # fn functio(db: &DbConn) -> Result<(), DbErr> {
509//! # use sea_orm::{query::*, FromQueryResult, raw_sql};
510//! #[derive(FromQueryResult)]
511//! struct CakeWithBakery {
512//!     name: String,
513//!     #[sea_orm(nested)]
514//!     bakery: Option<Bakery>,
515//! }
516//!
517//! #[derive(FromQueryResult)]
518//! struct Bakery {
519//!     #[sea_orm(alias = "bakery_name")]
520//!     name: String,
521//! }
522//!
523//! let cake_ids = [2, 3, 4]; // expanded by the `..` operator
524//!
525//! // can use many APIs with raw SQL, including nested select
526//! let cake: Option<CakeWithBakery> = CakeWithBakery::find_by_statement(raw_sql!(
527//!     Sqlite,
528//!     r#"SELECT "cake"."name", "bakery"."name" AS "bakery_name"
529//!        FROM "cake"
530//!        LEFT JOIN "bakery" ON "cake"."bakery_id" = "bakery"."id"
531//!        WHERE "cake"."id" IN ({..cake_ids})"#
532//! ))
533//! .one(db)?;
534//! # Ok(())
535//! # }
536//! ```
537//!
538//! ## 🧭 Seaography: instant GraphQL API
539//!
540//! [Seaography](https://github.com/SeaQL/seaography) is a GraphQL framework built for SeaORM.
541//! Seaography allows you to build GraphQL resolvers quickly.
542//! With just a few commands, you can launch a fullly-featured GraphQL server from SeaORM entities,
543//! complete with filter, pagination, relational queries and mutations!
544//!
545//! Look at the [Seaography Example](https://github.com/SeaQL/sea-orm/tree/master/examples/seaography_example) to learn more.
546//!
547//! <img src="https://raw.githubusercontent.com/SeaQL/sea-orm/master/examples/seaography_example/Seaography%20example.png"/>
548//!
549//! ## πŸ–₯️ SeaORM Pro: Professional Admin Panel
550//!
551//! [SeaORM Pro](https://github.com/SeaQL/sea-orm-pro/) is an admin panel solution allowing you to quickly and easily launch an admin panel for your application - frontend development skills not required, but certainly nice to have!
552//!
553//! SeaORM Pro has been updated to support the latest features in SeaORM 2.0.
554//!
555//! Features:
556//!
557//! + Full CRUD
558//! + Built on React + GraphQL
559//! + Built-in GraphQL resolver
560//! + Customize the UI with TOML config
561//! + Role Based Access Control *(new in 2.0)*
562//!
563//! Read the [Getting Started](https://www.sea-ql.org/sea-orm-pro/docs/install-and-config/getting-started/) guide to learn more.
564//!
565//! ![](https://raw.githubusercontent.com/SeaQL/sea-orm/refs/heads/master/docs/sea-orm-pro-dark.png#gh-dark-mode-only)
566//! ![](https://raw.githubusercontent.com/SeaQL/sea-orm/refs/heads/master/docs/sea-orm-pro-light.png#gh-light-mode-only)
567//!
568//! ## SQL Server Support
569//!
570//! [SQL Server for SeaORM](https://www.sea-ql.org/SeaORM-X/) offers the same SeaORM API for MSSQL. We ported all test cases and examples, complemented by MSSQL specific documentation. If you are building enterprise software, you can [request commercial access](https://forms.office.com/r/1MuRPJmYBR). It is currently based on SeaORM 1.0, but we will offer free upgrade to existing users when SeaORM 2.0 is finalized.
571//!
572//! ## Releases
573//!
574//! SeaORM 2.0 has reached its release candidate phase. We'd love for you to try it out and help shape the final release by [sharing your feedback](https://github.com/SeaQL/sea-orm/discussions/).
575//!
576//! + [Change Log](https://github.com/SeaQL/sea-orm/tree/master/CHANGELOG.md)
577//!
578//! SeaORM 2.0 is shaping up to be our most significant release yet - with a few breaking changes, plenty of enhancements, and a clear focus on developer experience.
579//!
580//! + [A Sneak Peek at SeaORM 2.0](https://www.sea-ql.org/blog/2025-09-16-sea-orm-2.0/)
581//! + [SeaORM 2.0: A closer look](https://www.sea-ql.org/blog/2025-09-24-sea-orm-2.0/)
582//! + [Role Based Access Control in SeaORM 2.0](https://www.sea-ql.org/blog/2025-09-30-sea-orm-rbac/)
583//! + [Seaography 2.0: A Powerful and Extensible GraphQL Framework](https://www.sea-ql.org/blog/2025-10-08-seaography/)
584//! + [SeaORM 2.0: New Entity Format](https://www.sea-ql.org/blog/2025-10-20-sea-orm-2.0/)
585//! + [SeaORM 2.0: Entity First Workflow](https://www.sea-ql.org/blog/2025-10-30-sea-orm-2.0/)
586//! + [SeaORM 2.0: Strongly-Typed Column](https://www.sea-ql.org/blog/2025-11-11-sea-orm-2.0/)
587//! + [What's new in SeaORM Pro 2.0](https://www.sea-ql.org/blog/2025-11-21-whats-new-in-seaormpro-2.0/)
588//! + [SeaORM 2.0: Nested ActiveModel](https://www.sea-ql.org/blog/2025-11-25-sea-orm-2.0/)
589//! + [A walk-through of SeaORM 2.0](https://www.sea-ql.org/blog/2025-12-05-sea-orm-2.0/)
590//! + [How we made SeaORM synchronous](https://www.sea-ql.org/blog/2025-12-12-sea-orm-2.0/)
591//! + [SeaORM 2.0 Migration Guide](https://www.sea-ql.org/blog/2026-01-12-sea-orm-2.0/)
592//! + [SeaORM now supports Arrow & Parquet](https://www.sea-ql.org/blog/2026-02-22-sea-orm-arrow/)
593//! + [SeaORM 2.0 with SQL Server Support](https://www.sea-ql.org/blog/2026-02-25-sea-orm-x/)
594//!
595//! If you make extensive use of SeaQuery, we recommend checking out our blog post on SeaQuery 1.0 release:
596//!
597//! + [The road to SeaQuery 1.0](https://www.sea-ql.org/blog/2025-08-30-sea-query-1.0/)
598//!
599//! ## License
600//!
601//! Licensed under either of
602//!
603//! -   Apache License, Version 2.0
604//!     ([LICENSE-APACHE](LICENSE-APACHE) or <http://www.apache.org/licenses/LICENSE-2.0>)
605//! -   MIT license
606//!     ([LICENSE-MIT](LICENSE-MIT) or <http://opensource.org/licenses/MIT>)
607//!
608//! at your option.
609//!
610//! ## Contribution
611//!
612//! Unless you explicitly state otherwise, any contribution intentionally submitted
613//! for inclusion in the work by you, as defined in the Apache-2.0 license, shall be
614//! dual licensed as above, without any additional terms or conditions.
615//!
616//! We invite you to participate, contribute and together help build Rust's future.
617//!
618//! A big shout out to our contributors!
619//!
620//! [![Contributors](https://opencollective.com/sea-orm/contributors.svg?width=1000&button=false)](https://github.com/SeaQL/sea-orm/graphs/contributors)
621//!
622//! ## Who's using SeaORM?
623//!
624//! Here is a short list of awesome open source software built with SeaORM. Feel free to [submit yours](https://github.com/SeaQL/sea-orm/blob/master/COMMUNITY.md#built-with-seaorm)!
625//!
626//! | Project | GitHub | Tagline |
627//! |---------|--------|---------|
628//! | [Zed](https://github.com/zed-industries/zed) | ![GitHub stars](https://img.shields.io/github/stars/zed-industries/zed.svg?style=social) | A high-performance, multiplayer code editor |
629//! | [Servo](https://github.com/servo/servo) | ![GitHub stars](https://img.shields.io/github/stars/servo/servo.svg?style=social) | The Servo Parallel Browser Engine Project |
630//! | [OpenObserve](https://github.com/openobserve/openobserve) | ![GitHub stars](https://img.shields.io/github/stars/openobserve/openobserve.svg?style=social) | Open-source observability platform |
631//! | [RisingWave](https://github.com/risingwavelabs/risingwave) | ![GitHub stars](https://img.shields.io/github/stars/risingwavelabs/risingwave.svg?style=social) | Stream processing and management platform |
632//! | [Warpgate](https://github.com/warp-tech/warpgate) | ![GitHub stars](https://img.shields.io/github/stars/warp-tech/warpgate.svg?style=social) | Smart SSH bastion that works with any SSH client |
633//! | [LLDAP](https://github.com/nitnelave/lldap) | ![GitHub stars](https://img.shields.io/github/stars/nitnelave/lldap.svg?style=social) | A light LDAP server for user management |
634//! | [Svix](https://github.com/svix/svix-webhooks) | ![GitHub stars](https://img.shields.io/github/stars/svix/svix-webhooks.svg?style=social) | The enterprise ready webhooks service |
635//! | [Ryot](https://github.com/IgnisDa/ryot) | ![GitHub stars](https://img.shields.io/github/stars/ignisda/ryot.svg?style=social) | The only self hosted tracker you will ever need |
636//! | [OctoBase](https://github.com/toeverything/OctoBase) | ![GitHub stars](https://img.shields.io/github/stars/toeverything/OctoBase.svg?style=social) | A light-weight, scalable, offline collaborative data backend |
637//! | [System Initiative](https://github.com/systeminit/si) | ![GitHub stars](https://img.shields.io/github/stars/systeminit/si.svg?style=social) | DevOps Automation Platform |
638//!
639//! ## Sponsorship
640//!
641//! [SeaQL.org](https://www.sea-ql.org/) is an independent open-source organization run by passionate developers.
642//! If you feel generous, a small donation via [GitHub Sponsor](https://github.com/sponsors/SeaQL) will be greatly appreciated, and goes a long way towards sustaining the organization.
643//!
644//! ### Gold Sponsors
645//!
646//! <table><tr>
647//! <td><a href="https://qdx.co/">
648//!   <img src="https://www.sea-ql.org/static/sponsors/QDX.svg" width="138"/>
649//! </a></td>
650//! </tr></table>
651//!
652//! [QDX](https://qdx.co/) pioneers quantum dynamics-powered drug discovery, leveraging AI and supercomputing to accelerate molecular modeling.
653//! We're immensely grateful to QDX for sponsoring the development of SeaORM, the SQL toolkit that powers their data intensive applications.
654//!
655//! ### Silver Sponsors
656//!
657//! We're grateful to our silver sponsors: Digital Ocean, for sponsoring our servers. And JetBrains, for sponsoring our IDE.
658//!
659//! <table><tr>
660//! <td><a href="https://www.digitalocean.com/">
661//!   <img src="https://www.sea-ql.org/static/sponsors/DigitalOcean.svg" width="125">
662//! </a></td>
663//!
664//! <td><a href="https://www.jetbrains.com/">
665//!   <img src="https://www.sea-ql.org/static/sponsors/JetBrains.svg" width="125">
666//! </a></td>
667//! </tr></table>
668//!
669//! ## Mascot
670//!
671//! A friend of Ferris, Terres the hermit crab is the official mascot of SeaORM. His hobby is collecting shells.
672//!
673//! <img alt="Terres" src="https://www.sea-ql.org/SeaORM/img/Terres.png" width="400"/>
674//!
675//! ## πŸ¦€ Rustacean Sticker Pack
676//! The Rustacean Sticker Pack is the perfect way to express your passion for Rust. Our stickers are made with a premium water-resistant vinyl with a unique matte finish.
677//!
678//! Sticker Pack Contents:
679//!
680//! + Logo of SeaQL projects: SeaQL, SeaORM, SeaQuery, Seaography
681//! + Mascots: Ferris the Crab x 3, Terres the Hermit Crab
682//! + The Rustacean wordmark
683//!
684//! [Support SeaQL and get a Sticker Pack!](https://www.sea-ql.org/sticker-pack/) All proceeds contributes directly to the ongoing development of SeaQL projects.
685//!
686//! <a href="https://www.sea-ql.org/sticker-pack/"><img alt="Rustacean Sticker Pack by SeaQL" src="https://www.sea-ql.org/static/sticker-pack-1s.jpg" width="600"/></a>
687#![doc(
688    html_logo_url = "https://raw.githubusercontent.com/SeaQL/sea-query/master/docs/SeaQL icon dark.png"
689)]
690
691mod database;
692mod docs;
693mod driver;
694pub mod dynamic;
695/// Module for the Entity type and operations
696pub mod entity;
697/// Error types for all database operations
698pub mod error;
699/// This module performs execution of queries on a Model or ActiveModel
700mod executor;
701/// Types and methods to perform metric collection
702pub mod metric;
703/// Types and methods to perform queries
704pub mod query;
705#[cfg(feature = "rbac")]
706#[cfg_attr(docsrs, doc(cfg(feature = "rbac")))]
707pub mod rbac;
708/// Types that defines the schemas of an Entity
709pub mod schema;
710/// Helpers for working with Value
711pub mod value;
712
713#[doc(hidden)]
714#[cfg(all(feature = "macros", feature = "tests-cfg"))]
715pub mod tests_cfg;
716mod util;
717
718pub use database::*;
719#[allow(unused_imports)]
720pub use driver::*;
721pub use entity::*;
722pub use error::*;
723pub use executor::*;
724pub use query::*;
725pub use schema::*;
726
727#[cfg(feature = "macros")]
728pub use sea_orm_macros::{
729    DeriveActiveEnum, DeriveActiveModel, DeriveActiveModelBehavior, DeriveActiveModelEx,
730    DeriveArrowSchema, DeriveColumn, DeriveDisplay, DeriveEntity, DeriveEntityModel, DeriveIden,
731    DeriveIntoActiveModel, DeriveMigrationName, DeriveModel, DeriveModelEx, DerivePartialModel,
732    DerivePrimaryKey, DeriveRelatedEntity, DeriveRelation, DeriveValueType, FromJsonQueryResult,
733    FromQueryResult, raw_sql, sea_orm_compact_model as compact_model, sea_orm_model as model,
734};
735
736pub use sea_query;
737pub use sea_query::Iden;
738
739pub use sea_orm_macros::EnumIter;
740pub use strum;
741
742#[cfg(feature = "with-arrow")]
743pub use sea_orm_arrow::arrow;
744
745#[cfg(feature = "sqlx-dep")]
746pub use sqlx;