From 68b227716c4aacd225eaf4b34e32ab8eb90d0238 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Marko=20Milenkovi=C4=87?= Date: Tue, 19 Nov 2024 01:55:34 +0000 Subject: [PATCH] Update root `README.md` and other documentation with latest changes (#1113) * Update root `README.md` adding example how to use ... and cleanup some of the context. Relates to #1105 * add new architectural diagram * address code review comments * address review comments --- README.md | 85 +++++++----- .../source/contributors-guide/architecture.md | 2 +- .../ballista_architecture.excalidraw.svg | 11 ++ docs/source/user-guide/configs.md | 92 ++++++++----- .../user-guide/deployment/quick-start.md | 60 +++++---- docs/source/user-guide/faq.md | 2 +- docs/source/user-guide/flightsql.md | 2 + docs/source/user-guide/metrics.md | 2 + docs/source/user-guide/rust.md | 122 ++++++++++++------ docs/source/user-guide/scheduler.md | 2 + examples/README.md | 100 +++++++++----- 11 files changed, 316 insertions(+), 164 deletions(-) create mode 100644 docs/source/contributors-guide/ballista_architecture.excalidraw.svg diff --git a/README.md b/README.md index df30641ee..00936c734 100644 --- a/README.md +++ b/README.md @@ -17,29 +17,58 @@ under the License. --> -# Ballista: Distributed SQL Query Engine, built on Apache Arrow +# Ballista: Making DataFusion Applications Distributed -Ballista is a distributed SQL query engine powered by the Rust implementation of [Apache Arrow][arrow] and -[Apache Arrow DataFusion][datafusion]. +Ballista is a distributed execution engine which makes [Apache DataFusion](https://github.com/apache/datafusion) applications distributed. -If you are looking for documentation for a released version of Ballista, please refer to the -[Ballista User Guide][user-guide]. +Existing DataFusion application: -## Overview +```rust +use datafusion::prelude::*; -Ballista implements a similar design to Apache Spark (particularly Spark SQL), but there are some key differences: +#[tokio::main] +async fn main() -> datafusion::error::Result<()> { + let ctx = SessionContext::new(); -- The choice of Rust as the main execution language avoids the overhead of GC pauses and results in deterministic - processing times. -- Ballista is designed from the ground up to use columnar data, enabling a number of efficiencies such as vectorized - processing (SIMD) and efficient compression. Although Spark does have some columnar support, it is still - largely row-based today. -- The combination of Rust and Arrow provides excellent memory efficiency and memory usage can be 5x - 10x lower than - Apache Spark in some cases, which means that more processing can fit on a single node, reducing the overhead of - distributed compute. -- The use of Apache Arrow as the memory model and network protocol means that data can be exchanged efficiently between - executors using the [Flight Protocol][flight], and between clients and schedulers/executors using the - [Flight SQL Protocol][flight-sql] + // register the table + ctx.register_csv("example", "tests/data/example.csv", CsvReadOptions::new()).await?; + + // create a plan to run a SQL query + let df = ctx.sql("SELECT a, MIN(b) FROM example WHERE a <= b GROUP BY a LIMIT 100").await?; + + // execute and print results + df.show().await?; + Ok(()) +} +``` + +can be distributed with few lines of code changed: + +> [!IMPORTANT] +> There is a gap between DataFusion and Ballista, which may bring incompatibilities. The community is working hard to close this gap + +```rust +use ballista::prelude::*; +use datafusion::prelude::*; + +#[tokio::main] +async fn main() -> datafusion::error::Result<()> { + // create DataFusion SessionContext with ballista standalone cluster started + let ctx = SessionContext::standalone(); + + // register the table + ctx.register_csv("example", "tests/data/example.csv", CsvReadOptions::new()).await?; + + // create a plan to run a SQL query + let df = ctx.sql("SELECT a, MIN(b) FROM example WHERE a <= b GROUP BY a LIMIT 100").await?; + + // execute and print results + df.show().await?; + Ok(()) +} +``` + +If you are looking for documentation or more examples, please refer to the [Ballista User Guide][user-guide]. ## Architecture @@ -51,19 +80,10 @@ can be run as native binaries and are also available as Docker Images, which can The following diagram shows the interaction between clients and the scheduler for submitting jobs, and the interaction between the executor(s) and the scheduler for fetching tasks and reporting task status. -![Ballista Cluster Diagram](docs/source/contributors-guide/ballista.drawio.png) +![Ballista Cluster Diagram](docs/source/contributors-guide/ballista_architecture.excalidraw.svg) See the [architecture guide](docs/source/contributors-guide/architecture.md) for more details. -## Features - -- Supports cloud object stores. S3 is supported today and GCS and Azure support is planned. -- DataFrame and SQL APIs available from Python and Rust. -- Clients can connect to a Ballista cluster using [Flight SQL][flight-sql]. -- JDBC support via Arrow Flight SQL JDBC Driver -- Scheduler REST UI for monitoring query progress and viewing query plans and metrics. -- Support for Docker, Docker Compose, and Kubernetes deployment, as well as manual deployment on bare metal. - ## Performance We run some simple benchmarks comparing Ballista with Apache Spark to track progress with performance optimizations. @@ -81,19 +101,14 @@ that, refer to the [Getting Started Guide](ballista/client/README.md). ## Project Status -Ballista supports a wide range of SQL, including CTEs, Joins, and Subqueries and can execute complex queries at scale. +Ballista supports a wide range of SQL, including CTEs, Joins, and subqueries and can execute complex queries at scale, +but still there is a gap between DataFusion and Ballista which we want to bridge in near future. Refer to the [DataFusion SQL Reference](https://datafusion.apache.org/user-guide/sql/index.html) for more information on supported SQL. -Ballista is maturing quickly and is now working towards being production ready. See the [roadmap](ROADMAP.md) for more details. - ## Contribution Guide Please see the [Contribution Guide](CONTRIBUTING.md) for information about contributing to Ballista. -[arrow]: https://arrow.apache.org/ -[datafusion]: https://github.com/apache/arrow-datafusion -[flight]: https://arrow.apache.org/blog/2019/10/13/introducing-arrow-flight/ -[flight-sql]: https://arrow.apache.org/blog/2022/02/16/introducing-arrow-flight-sql/ [user-guide]: https://datafusion.apache.org/ballista/ diff --git a/docs/source/contributors-guide/architecture.md b/docs/source/contributors-guide/architecture.md index 076f5ab97..5f0333f9b 100644 --- a/docs/source/contributors-guide/architecture.md +++ b/docs/source/contributors-guide/architecture.md @@ -65,7 +65,7 @@ can be run as native binaries and are also available as Docker Images, which can The following diagram shows the interaction between clients and the scheduler for submitting jobs, and the interaction between the executor(s) and the scheduler for fetching tasks and reporting task status. -![Ballista Cluster Diagram](ballista.drawio.png) +![Ballista Cluster Diagram](ballista_architecture.excalidraw.svg) ### Scheduler diff --git a/docs/source/contributors-guide/ballista_architecture.excalidraw.svg b/docs/source/contributors-guide/ballista_architecture.excalidraw.svg new file mode 100644 index 000000000..a57e85367 --- /dev/null +++ b/docs/source/contributors-guide/ballista_architecture.excalidraw.svg @@ -0,0 +1,11 @@ + + + eyJ2ZXJzaW9uIjoiMSIsImVuY29kaW5nIjoiYnN0cmluZyIsImNvbXByZXNzZWQiOnRydWUsImVuY29kZWQiOiJ4nO1daVNcItuy/X5+hdHvy3snrnX3PNxvooiKolxijq9vXHUwMDE4XHUwMDA1XHUwMDE0g5aAXGaKvrj//WXS3VJcdTAwMDVVUMogdkvE6e7DULWrKtfamblX5v6/vzY2vvWe2963f21881x1MDAwNmXXb1Q67tO3f+D7j16n22g14SM2/P9uq98pXHUwMDBmv1nv9drdf/3zn6NfOOXW/Y9feb537zV7Xfje/8L/b2z83/DPwHk6XrnnNmu+N/zB8KPAqShcdTAwMWJ/N9dqXHUwMDBlT0utJFZcdTAwMTlm6es3XHUwMDFhzYo3wGOW6NboaI3uXHUwMDBljKLnVeCTqut3vdEn+Na3TPryzivvPewxpVx1MDAxZsvq/qnQuDhcdTAwMWH9vNrw/ULv2Vx1MDAxZlx1MDAwZbbbglx1MDAwYlx1MDAxY33W7XVad95Fo9Kr44jG3o/7VafVr9WbXrdcdTAwMWL6Tavtllx1MDAxYr1nfI+Q13d/3Jl/bYzewVx1MDAwYlRKvf7/8Fx1MDAxN1qMnXu75bc6eO7/olbTMlx1MDAxYp295JbvajCEZuX1O72O2+y23Vx1MDAwZTyo0feefl6VYMaxwdfrN+peo1bvXHK/Mnqz61xybzQnRlAqX9/GM7b3K0M7+Pfo3nbce29cdTAwMWa/3+z7fvBcdTAwMDY1Kz9v0C97XHUwMDE5WVxm//nOf0aXhN9PXHUwMDA3LG10hn674v549FRzqohlRFszuut+o3k3fnq/Vb5cdTAwMWJZy1+Bc73NeOGMccarueZMRlx1MDAxOW4queFcdTAwMWU17vqPqpi/7J1kts4u872t2+fzXHUwMDE4w6275Xq/43286eqAqeIvXHUwMDE4XHUwMDFiP/ebTPe/POuVveqk2TKhJlxylVI6bqhUaCk0I5Kun61cdTAwMTImjFx1MDAxNCuxVcZVnK1SaqhkRlNcdTAwMWRlr9vJ7bVwpbvdc5V7aPCT4yZ5bOQuL/31t1dcdTAwMTOyVy6UI+ey2Fx1MDAxMquyUun9XHUwMDE2qy21QivC/2yDXHUwMDE1sVx1MDAwNsuGw5Da2CiD3UlusGcnXHUwMDE3tefqyUEuf3Wb7tvHg1QmW1xce4M1JGSwQpM5XHK2WvXK1r7fYC2xXFxJzdbQXHUwMDFiWKHBUlx1MDAxNuvKamq45ZHWmk5urVx1MDAxN51OZYfsnKco8UvP4m73ort9tt5+LCVcImysXFyyWFP1XGKnnM4y1bItR5gqXHUwMDE1XHUwMDExbquS45bKXGI4iIworj/IVF9/M/r16/O9k1fl1LMmO3dpdkCLNeXbnb3XS1xyXHUwMDE5otvptJ6+vX7yn5//mlxuXHUwMDA0o1x1MDAwMny5VOa2U9xiarQwKjKk201cdTAwMGWFp+tWcb9+o2ul45Q5ylSN6T7l1lx1MDAxYlxu0lx1MDAwNml66Fx1MDAxOVx1MDAwYjMncbuyYqpcdTAwMTG+cUI0UHhcdTAwMTDMWC4+yjWehlx1MDAwNmNbmVx1MDAxZFPq7ZxcdTAwMTfq9bTW+9fbVi1cblxy4EFcdTAwMTCxXHUwMDFhNHBcIuLQwFx1MDAwNZNWc1x1MDAxOel3Z5Kjoan7XHUwMDBmxeuTcmG3cfKQ3r7P5nleLFx1MDAxNFxyXHUwMDE1t1v3XHUwMDE2XGZcdTAwMDc2XHUwMDA2XHUwMDA3zuZ1vD1dlXPAXHUwMDAxI3YjmVrHuaHeXHUwMDE4mM30LU1cdTAwMWZcXNykWuePfcvTJ59cdTAwMTFcclwi1kniwnKjiYnMmuwnR0P2pv9UKOaKjZt0pV/eub16eczfrj1cdTAwMWHEXHUwMDE4XHUwMDFhXHUwMDA04Vx1MDAxZotcdTAwMDaurFx1MDAxNVxcyDWMQldpryrel9GcSys5j7LXbHJ7LW7d1OTVo85kintnz/3inrnR7bW3Vz5ur4Z9rL0qJinTlKxhmm+x9trzXHUwMDA2vShTjfe6XHUwMDE5sUJcdTAwMTLOSWRcdTAwMDR6mNxU5cC7yb/kXHUwMDA2+5e3/tbR7v3xfVvdrbfbbYR22LjfXHUwMDFkXHUwMDFmg87ldVPqXHUwMDE4ba3VTIJPra2etFs9YbdcdTAwMDaCXsOtWkDyJPTBUrMko1G1mr1C4+WHlYXe3XXvXHUwMDFiPt7u0YVcci1cdTAwMTfu4E5cdTAwMDNue6PUhzF8b+b7Xuf5e/PEd5tNr/Mt9OUtv1FD0/5WhotcYn5cYlx1MDAwNt5rlF3/9Vx1MDAwYr1WgDLLcHq3XHUwMDAxR5u8Xa1Oo9Zoun7xLUNx+73Wqdf9cZm9Tt9cdTAwMGLeKW/vNTNcdTAwMDaG9j7k6vE3fyHXKousXHUwMDE2XHUwMDE5IFx1MDAxYyXH7eb9qdt7clutzS5cdTAwMDVcdTAwMGI/ITbFntY9z2kk3FBcXI+UgjGYh0f0NoxcdTAwMTa0XHUwMDFkXHUwMDFiyWKWlShcdTAwMTFcdTAwMGXFsypcdTAwMGJTmmA0YvKRo4H+XHUwMDA0seCKXHUwMDE4Q8VcdTAwMDJCh89cdTAwMDLi9MAr93tgq1x1MDAxYt+bmY7brndXjd1pI1g2ZFVsUE9cdFx1MDAwNDKEXHUwMDEyXHUwMDE1OdnmkoP26aiUhYD0WpOD5+Yjv7zdp7S09qBV4dU0SdXYuVx1MDAxN7M0oakjXHIhhML0SaRQXHUwMDExKGWTU61cdTAwMDE25UQsIKT5LCg98mB6K8dB0/eqvSVcdTAwMDBz4qTLRiNV8SpcIiWJNsZEz6HHb1xi01x1MDAwZU82K/TlrrJ/3d8/eG61XHUwMDFiNp1edziqwJr+0PXV82ZcdTAwMTXiXHUwMDE2tylhXHUwMDBlN1x1MDAwMoiPQ8Am6MjER4CcmDapXHUwMDEwXHUwMDE2uFJ9JCDfXHUwMDFjnEVcdTAwMDEywHIzXHUwMDAwWTs92d4oeJ3HRtlbLSqjz7x0aFx1MDAwNlx1MDAxNp3GoGkkMDfTJHIx6CQ5Mu3uVvplkD+uvVx1MDAxY1x1MDAxNIgqp3gus7n2q/gqcNk/XHUwMDE2RvnYuVx1MDAxN4RLrlx1MDAxY1xiTNGNplRTXHUwMDEzXGJAp1x1MDAwMJNrw1xmN3ZcdTAwMDHJlOnA/PnBJ8h5z1x1MDAwN/tdXHUwMDFm7/VGIX+4sXWy//ffq4V+/NmXXHJ/xmPhT1x1MDAxObNMiFx1MDAxONlZPjn+72/kpp++PehcXJVe7O1NYd+9zK+97Ewx6nAhwIPVXHUwMDFh/rBj2VS2rFmaOEoqQqSRlifjXHUwMDAySo3gXHUwMDEyfKiPzFCtXHUwMDE4rqfpQvEjgFx1MDAxYXXeZUOUXHUwMDBiOf7uL4hKxSBk0tFcdTAwMGLUhTfkjfP0wFx1MDAxZVdfTkui0k5tV2pcIu96n1x1MDAwMaFSaWshelCUmLCsXHUwMDE5ooolIVQ7nEFoXHUwMDBi5GhVwulcdTAwMWEuhlx1MDAxOFxidOjS5+v1gehJp3Xv9epev7tqkEafeemOtDDj745iXFxBmVx1MDAwMlx1MDAwYo1cXDovJsfp5Vx1MDAwMS9cdTAwMTXuje56LJM6XHUwMDE5nJztnFxydtdcdTAwMWSnXHUwMDE0JjPHXHUwMDEwXHUwMDAwqjAwb1x1MDAwNtVnP1x1MDAxMsXxXHUwMDE5qCSKw8VcdTAwMDa8iivCzcdGvIQzXHUwMDE1kGV+RbzvXHUwMDAz6nTdgI5HK8VqXHUwMDE0IaNXY8+So7V8uXN317woXHUwMDFmZs+vjlx1MDAxYuDjX9vduFWdcqfV7W7W3V65/vGI5VSPSyHlvFx1MDAxYfbYRVk9Slxyvlx1MDAwMtOQcMXbRHVcdTAwMWIlXG5cdTAwMWWQXGZKXHUwMDBm1kdcdTAwMGL2iZSRsWGhja9GgniIaS1FJDzOk8Mjk637L97WUXHrYquW3qvyQfVyf90nM67plLmMqflcdTAwMTY94+YyxVx1MDAxZMMwXHUwMDE25Vx1MDAxMJRLaVwiyj7E5FxcJlx1MDAxNdiMsnpcdTAwMDFcdTAwMTj5LE5nXHUwMDAxhlx1MDAwYrb6vblcct9cdTAwMGLZ9irms/izr2ROXHUwMDEzNFanXHUwMDAwXHUwMDBlXHLXxMaUvF4kXHUwMDA3bfXwmbv+Qzmdq25uXHUwMDBmrp9cdTAwMDat/n5cXMnres1p3I7rmdU0eX+VmDIh71v8TKiIs8RwLciXnnm+OezH+SPwIHX8JCZcdTAwMDW3xkbD4So5XHUwMDFjpk/1a1xmXHUwMDA3MVntYpijXHUwMDE0XHUwMDE1XHUwMDEyo1x1MDAxZrgzbD5hQHxYRp2InIlcdTAwMTg790RcciMjXHUwMDE0/Fx1MDAwZa74UopcdTAwMTjZbLRMt2dcdTAwMWFIN71hkuv23E4vXHUwMDA1ttdo1sJcdTAwMDP72bdjP0EkMZxcdTAwMTbLfVx1MDAxY+UmcYhcdTAwMDVcdTAwMDdNMSO5hvA38Fx1MDAxOPGOue2gxVxmLXTgVU5ajWbv54gnboTXrMxcdTAwMWXg9HqvwFx1MDAwMIlDtTJCMVxi+1x0weKmecfnu93eduv+vtHrhb9cdTAwMTm+yVvIXHUwMDEydc+dMFxmuL7gZ996ncbYXGbbxoOGXHUwMDE5c/SvjVx1MDAxMbCG//P673//I/LbIfvH1+aE6Y9cdTAwMGUxQaWeX2o9zc2LJlZcdTAwMWMlKCfwilx1MDAxNmNcXCfnxenTx3rzYthJsNLhilx1MDAwMciF1UKJ+EXgJFx1MDAwZUNcdTAwMWMpRvV04SZ04slcdTAwMGUvSitNlTZcdTAwMWapaVxcOvPpQvtmp9hxaeWqeL374m8+XHUwMDE0O6lI5qOOXHUwMDE2gnBhrCSEk8CwXolFOTDnayO0ooxcdTAwMThgn8VcdTAwMTPhdK1DYLzUgVx0zWjKlSTKcjpah1x1MDAxZPEge9P4Plx1MDAxOVx1MDAxMVpcdTAwMWL8+uaEuS+IXHUwMDA1p0dMXHUwMDAxP2OMXHSVtsxcdTAwMThcdTAwMWVcYrVcdTAwMDNMWErOhJmty9btMaM6f2hOXHUwMDA2V6xaS+/GLa2tSUlcdTAwMDbYJHekplx1MDAwMpxcYvxzxDw/Wl1RR1x1MDAwNFJzZi5dzPTOV9qG84BcdTAwMTE5fFx1MDAxMdCdvyZcdTAwMDdcclx1MDAxMDVMZetVasRcdTAwMDRcblx0tJVcdTAwMDEtwTJL40S8sotKoyRjOro2rvSWJFx1MDAxZTk8bJxcdTAwMWZ66U725bHbKlx1MDAxZlx1MDAxNHdujtfcvFFkaX/aNv5cdTAwMTlu5cYld5RcdTAwMTh9XHUwMDFja90raIdcdTAwMDFhquZaM1x1MDAxYp3CXHUwMDFi2ebAXHUwMDFlXHUwMDFm7e9svpyXSe/6bNPdur995L9i8t+3a4bRXHUwMDA00LT0bLiMb3VEYVx1MDAwMMRSXHUwMDE2vbRbTlx1MDAwZaRrXVx1MDAxYmw/XHUwMDBmXHUwMDA2/kOh/nh45vJ2o7X28mXKLEBJXHUwMDEyI6mFgNOMdUVcdTAwMDTTdUxgXCKZXHUwMDBmS1x1MDAwYl3olYQyzoJUvGxgvdfHXHUwMDFl2rj6Wlx1MDAwZV5q6lxcyfi5XHUwMDEyXHUwMDA1XHUwMDAxmKyInCu95Fxif9h8qJxcdTAwMTdfzu3V7rFcdTAwMTZcdTAwMDeN7tnJlrv2XGJcdTAwMTfCxvuC4K8n9Vx1MDAwNVnVekK8XHUwMDBm4ZzRmX4g5cwhwdeEW2g4irBIzFxc+pFcdTAwMDXoRlx1MDAxOc6JXHRcdTAwMWH2MqYxbmM1XHUwMDBmilx1MDAwYoi6gkuFXHUwMDAxXHUwMDFirya38e5ZPt+0J4fd3uBp+ylz/Pzks7Vf01x1MDAwNS+LOVx1MDAxMNZAzKA0zCBibJ2Ihq1vvoAnvrCVU0fNqFx1MDAwM5hsXHUwMDAywpjhVPJcdTAwMGZcXOJcdTAwMDXjNVK+xXij5qfkXHUwMDE1c8NcIvCNQs+teSsum4s+8/JcdTAwMGJ04ld1sW2hIDFcbv1acthmy5WzZqfQKeayXjUr7navO+ThM6Rr7ViVXHUwMDBl3Kxl1c9B0MhgWmHAk9KwwPQyXHUwMDA1nlx1MDAxNFx1MDAxM1mSc/1cdTAwMDdVtKZcXN9vdHvuRqFcXPcqfT+2Z8SSUDrt/MvGqphcIptSXFxCmCjoiClcdTAwMDNYrSfHKlx1MDAxZPBmYT8vM+X62cUj2Vx1MDAxM0+Dh+fPgFVcdTAwMDCDdVx1MDAwMIFcdTAwMTChXHUwMDE5XHUwMDBlXHUwMDExo1x0l75Sq1x1MDAxNphXnIJj6qBcdTAwMWOfKIhcdTAwMTcllSpCflx1MDAxOFx1MDAwMWRwXHUwMDFkjVDcftBiy/tyh1x1MDAwYlx1MDAwMvKPPlx1MDAwZa2PwvHk6VdcdTAwMTJcdTAwMTJqXHUwMDFh30LCXG5tXGJcdTAwMGIsXHUwMDE1XHUwMDA2oNxIXHUwMDBlZb19fNn0Tve6pTN6NMg83PrtYj5cdTAwMDbK6+Mtc0PjI0LBeVx1MDAxOMVzgbhqqrbqToJYXGJcdTAwMWRcdTAwMGX2ovpgS1x1MDAxNlx1MDAxNzT+XG5cYlFOQlx1MDAxNlFDu/Ck6e1cdTAwMTbtsJt0Jf1ynMu7zztNclhcdTAwMWNcdTAwMTlWyIjflDRd9Vx1MDAxYYSksVx1MDAwYmxcdTAwMTR7eVx1MDAwMlx1MDAwN+tIXHUwMDE03b5hQmxvqWrRd7Ptx+tylnX6XHUwMDE3Rd5Ze1x1MDAxNFx0xlx1MDAxY1x1MDAxMotcIqtcdTAwMTc5XHUwMDE3zqtTVFx1MDAxNlx1MDAxN6hcdTAwMTehI144UHStf97Te/Jxd5dl3WylV2z55DNcdTAwMDJFTNHuXHUwMDFhXHJAMYZFXHUwMDAx5S45UPxdm8letUpcdTAwMGY2I3yayVx1MDAxZZ7TVHPtgaJcdTAwMDT/LEBRTEvClV5HQa9bu1x1MDAxM43dwoVu1p/22+e9s2t+ml9PoMQuw7FYjHBcYq2otdGJXHUwMDEwPzlE7lx1MDAwZki2XWmWRO5l+7L9xI9cdTAwMGJNWV17iFxiKVx1MDAxZIndgCmWXHUwMDE4U1x1MDAxNa6EXHUwMDE2cpGCjfj9nqx1cDtcdC5cdTAwMDTjhlx1MDAwNbb4mlZvqakxmHv+yFx1MDAwNCbW5lx1MDAwNtbHlr3AtuP23GpcdTAwMWatd2PUIe+/i273buNn3cj/rDbSetOAlp1CkYGlzonKM66VoipanXqfXHUwMDFj5c+kV99q1vZ1p+Vtq7uz3GXjpbH+KFx1MDAxN8ZcdTAwMDEnXGbQJSi3MuBcdTAwMGVcZu1sbI1svsArVqjKXHUwMDFkXHUwMDA1VG+ppIpSPlJQTlx1MDAwM7mQWklBPrA9iVx1MDAwMWJcbubmll6GVu9Xq753ikrKlUI58sRLh2z8ulwitWAnglx1MDAwNPv/XHUwMDA1INtMXHUwMDBl2W6/WDu52K5cdTAwMTcvty46p8V6+eEgp9dcdTAwMWWySllHXHUwMDAzXlx1MDAwMSiUc/hXXHUwMDE4stSsXHUwMDAyssKx2MeIYPJZkMh851x1MDAwNGSVIFxmmHZcdTAwMTF7Mn5suvPNmL3oNHor1r1En3lRqI0tXHUwMDAzUVN65FpsKSVNIHtcdTAwMTOAbSs5bKeL/9ZcdTAwMTW21IY9aG5cdTAwMDBDgFx1MDAwNy5cdTAwMDBcdTAwMTFKmPGhvE3HXHUwMDE211x0QYuQKlVG5DRRbVx1MDAwM1x1MDAxM5pQuEmXVSzotr627JSCUqM+suj7zXuZXHUwMDA17meiqpDpO+1thOvhXGInWln280+tR2a/8avSQlx1MDAwNn+6mEKQ6cLowFx1MDAxMJcxwneUgkSUe6y4ICTW/oe/nrT80fH+XHUwMDFhO+7iXG5E9Fx1MDAxNPGFhfnbwCQ14opcdTAwMDBHtpNzZG4z1axuX1x1MDAxZe4+leTpw/NW/dw7XFx0NLL4XHUwMDFkZiiKe2PzcoqzxS1cdTAwMDNNr1x1MDAxMVnIUlx1MDAxMMOaR4hr5HptkbqE5Fx1MDAxYTpcdTAwMGbRXG7Y+O1RjSSGXHUwMDFiXHUwMDEyLV14SG7pN42z1PNpVz8zmupcdTAwMTfF3SFN+evvxFsmXHUwMDFkyiQws1x1MDAwMkOXOqxcdTAwMGVcdTAwMTRcbvtcdTAwMDFKotSwi7mYx9Bjc2uaOuCtXHUwMDAx31x1MDAxMK6lXG42XHUwMDFmfbVxi4BcdTAwMTOCa1x1MDAwM1GHoFx1MDAxM9Wi1OBcdTAwMDBMtEM/stj+826/99I9yNbr/m320Dy0SjrY9+Hb4cWVn3ngXHJV6O1cdTAwMWZuXHUwMDE3qpXjbKtcdTAwMWX8wuNjxmX5jr9/1yp0ztxcdTAwMTevv+vfLqhcdTAwMTTl3WX6c1x1MDAwNlx1MDAxNGNuyXjZZMwnXHUwMDBiLshcZn220JmXOLhcdTAwMTKO+9iAdUHAXHUwMDFlXHUwMDEwxeBcdTAwMGKCWEtcdTAwMTRMdPAx3ERqZ1x1MDAxZo9Q7PpnXGZH0GjJR3I5fGFZiJJcXFtccl41gfh41lx1MDAwMTHrhPu3UC0lhLGjqk18XHUwMDE56nBcIqiRVkmAXHUwMDAwobOORpUjiVx1MDAwNLwqRISmSoZcdTAwMGUnXHUwMDFjXHUwMDA2MGdcdTAwMTZLr8DV4DNcdTAwMDdnXHUwMDFjgFx1MDAxYzi+knFLwsdC5lx1MDAwMK9OWVx1MDAwNZFcdTAwMDNcdTAwMTEzj4WdXHUwMDEy4Vx0MFx1MDAwZTdPWcPDXHUwMDE3XG63QXNcdTAwMDO3TFx1MDAxMYk79c46XHUwMDFhPDicnO2w9yQ3NDw2eOpcdTAwMTT3dCRcdTAwMTKGTWY+XHUwMDAzLPsk2ip4qISpwJ5Hw1x1MDAwZrXD4FZZeJqWwm2Y+VxmkNGYXHUwMDA0f1eDgcDfJnQ04oDt4D5dyjDL9MzbhmOTwsD1wEwlXHUwMDE0XHS5jlx1MDAxMm6pgCdpLNiPhVx1MDAxOC7B0YhcdTAwMDIuNUilhlx1MDAxMFx1MDAxMTxcdTAwMWFEXHUwMDE0TOMuYuDiwO3DuqeZXHUwMDA3VMpcdTAwMTFobmBPYKRcIvRQN7mjKVx1MDAxM5aB+y+lXHUwMDE1duZTlcpcdTAwMDGgWoJ1gYAtXHUwMDE2gtamdPRcdTAwMTB88Fx1MDAxY4Y7l820XoFcdTAwMTldXHUwMDAznlx1MDAxY4S9uHZcdTAwMTM6XHUwMDFhzG34gFxieFx1MDAwMFx1MDAxNL3vmdBcdTAwMDdsYXZYMVx1MDAwM0ag4Hihwyk4XHUwMDFjXCKFXHRgXHUwMDE5uLUzXHUwMDBmx+FJwGVcbitx9mOchy9cdTAwMTbdOnhcdTAwMDZwvVx1MDAxNKyFs9lWjFx1MDAwMSHHXHUwMDE4XHUwMDAyi0GIQtfV6rHnS1x1MDAwNLiDwDJcdTAwMDBcZlx1MDAwNJuezXcxbPtX8O84h8zz/Ua7XHUwMDFiXHUwMDFkfXBcdTAwMTm/XHUwMDE4XHUwMDAyYFO491aUT9ZJ7pPVMtf181x1MDAxYyl23HT9fO/yeLvcTMXtbrlePlx1MDAxOdNYpqtxVSgsXHUwMDAyhyfnUDAvmFx1MDAwZpjEXjPL8snA3lx1MDAwMFx1MDAxOFx1MDAwNOzSmiifjFxuoEzsVMmwvIdcdTAwMDVcdTAwMWPsV6dMXGJcbvji0frw9XTLVpfQnYqbSsO9bzUrUbhBXHUwMDE3Jlx1MDAwZTcwhymYXHUwMDA3dGQs031DLGNt7d5cdTAwMWRcYr55vPWSvXq+vuBH2fXHXHJEKzBHQzRHLZCmXHUwMDFhU9OYYXNLTqzBSjlp5iuliFVcbjBcdTAwMDfLXG6x07LCRlx1MDAwNFx1MDAxMdtwXHUwMDAyvLWwgHJwhSxMm1x1MDAxM8V8XG7iMWLsLNz8ebCYMp3AXHUwMDA0XHUwMDFhv1xyXHUwMDE33mdcdTAwMDVOdCQueslxsd+9PjS59lx1MDAxNVx1MDAxY65116tdZvRdff1FZlbCfIKzXHTuqVx1MDAwMyFGeEKRjDhcdTAwMTBcdTAwMGUgeiTMuXS+hbpYWIB7b1BpKbUgjJmoOnbmgOGDy1x1MDAwM4FcdTAwMGU4L0xNpP9cdTAwMTnBLjrMxGjQ/mRkxGW+sH4yXHUwMDE2XHUwMDE2VECUXG5GXHUwMDExKVLuJ4fF7oOXPslcdHNZK2UuSTprdlx1MDAxYqm4XG67NYKFXHUwMDEwXHUwMDBlh5hFKFx1MDAwMc4wXHUwMDFi65RcdTAwMDJeuUOHdTGAXHUwMDBiwyVdTvJcdTAwMGK3+Vx1MDAwMEtQQE+CQVxcXHUwMDE2UXlHraNcdTAwMTRcdTAwMDTaQqJcdTAwMTCQT05cdTAwMTeUQFx1MDAwMIhcdTAwMTneP1x1MDAxMlx1MDAxNoHH8lvnsShxhIHQXHUwMDBlOyliXXIoXHIwYSOzjrZJKcSKuJ5rNVx1MDAwNp9Sj1x1MDAxZE4wylx1MDAxOcTgXHUwMDE2o+eZgfZcXEFibHEwUfFcdTAwMTWHhjBcctzGI7nrMTl3nWerorq1c89cdTAwMWVEZ6uXfm7SXHUwMDFktf5cdTAwMDVcdTAwMTbGXG7gXHLJMD5kPJy0l1I7XHUwMDEyk2yUMMxGLMfPpZw7TEpcdTAwMDZwXHUwMDE14GxHllx1MDAxYWrwLIDZMFx1MDAxNYGGqycndFxublx1MDAwN3bYmUFcXJ+al4KCXHUwMDFkfDLw8CBq1phcdTAwMDBcdTAwMTWhL73qd0ZG8Uu/c1xcuvXKvY1Cr9Vxa9735nHne3O34XtcdTAwMWKF527PXHUwMDBiqENXs/l14uEk1vlwXHUwMDE5XFzmpFP5YkbXQ1x1MDAxZK/8keBcdTAwMGJcdTAwMTMto/tcdTAwMWU+vSFcdTAwMGWgvduDXFzh+vb06url3ofb8PByXHUwMDEwQ1x1MDAxYVNcdTAwMWLDjWdvlqyjx8SuXHUwMDAwc8aunOFcdTAwMDJlhf3iNaVcZlx1MDAxYq9ibnou0qhWq7ZcdTAwMTIh15NcdTAwMTY9LoXtkVx1MDAxOYRcdTAwMDMywtuxmN5cdTAwMTLgkVx1MDAwMbVJnOvGOUNSRqigMV1cdTAwMDdGnHHuiXb+2Vx1MDAxY5y+kNblzY2q0PbJ5Zpzxlx1MDAxY2ZcdTAwMWbUX05W50PYZIiNXGZ/XHUwMDA3yc3+7PCg5dtcXLu9f0PPKC3tXHUwMDFmNfYy6272XHUwMDAyXFwjXFw+sVx1MDAxOH8yNm72XHUwMDEwdIK1Y9hpObPzZVOXafbUKI4rgWzWXFyZu5SP9/rkfu+4dtfffc7v7LdcdTAwMWG1oJP/h1x1MDAwMWNKXojDrCOjPcjn5Kg4rFx1MDAxNdj22daO6W/uXHUwMDE3Ui9cdTAwMGZPu17vXV1CV4pcbupobL+nwTdcdTAwMDNiXGJLQlx1MDAxNUxcdTAwMTXgWErLXGauOZr1RYVcdTAwMDIvl1lcdTAwMWJTYbV6m3894+eu5o3NJtn47VqplCjj0dHVXHUwMDEwL8nxNCjtXHUwMDFjdfyjbn/Qtzf9otpRJ7Wd9+BppUWK1DGcYbGdsiZcdTAwMTjpXGbxZJVcdTAwMDP/gVx1MDAxOVx1MDAxYku1XHUwMDE2y8FcdTAwMTO3XHUwMDBlxVx1MDAwNW1cdTAwMDbAxcxeVIpcdTAwMTVgjWJcdTAwMTeYXHUwMDA3Ka6xXHUwMDA3t5H9Nc1wo6Umq1x1MDAwM9SXXHUwMDEwakVcdCRcYtgh2Fx1MDAwNvdcdTAwMTnJlDNcdTAwMTlcdTAwMTLMXHUwMDAw0XLGwFmCOyi5plx0lEFx5oavTeZILLNcdTAwMTBcdTAwMTbiXa6JXGZoXHUwMDE3/lxu/v12XHUwMDA2XG6sh0+UUKI+h1x1MDAwNLNGI1x1MDAwNmIkOVx1MDAwM/XqrNPdPM/lwI2yl+f3hfPqmV17XHUwMDA24lx1MDAwZbfEolDIUjvWf0pp4jBpmGRcdTAwMWO1XHUwMDFmfN5OXHUwMDAyMVxmZFx1MDAxY2ZcdTAwMTS29Fx1MDAwNSSi41x1MDAxNSFXZlx1MDAxMOqjJotcbsaJkTyiw73SXGJhbr446LfjICpcdTAwMWMqYFxu4lx1MDAwMqZCXHUwMDBiSFxySeM2J41jJlx1MDAwYsWa3PCAqGdcdTAwMDNcdTAwMDZi4OczoYxcdOnZ5iSi+HZ4gjNcYiqDS0ZcdTAwMDFcdTAwMWWib4gsXHUwMDFl6le5WiGjilfb2f1qpmNPtt6VZlo1XHUwMDBmMfDLtTIwN/BRzlx1MDAwMVx1MDAwZlx1MDAwMKG2Y4xcdTAwMDBcdTAwMTOAp0a1nLODVlxmXHUwMDBmXHTiKFRLMql+YDGChzhMUFxufDXL4XNjXCJoiFx1MDAxMc10qP/oXHUwMDE3XHJt/Fx1MDAxZTSEXCJcdTAwMWSLglqFLW2stlx1MDAxM6xcdTAwMDFRp6RgpSitxWzjTFx1MDAxZaKOwq1cdTAwMTJwRUPTwFZa+KJcdTAwMGXETVx1MDAwMlx1MDAwNdRKgNMtzMzDxVvwcIBwMvDjsFx1MDAxZlx1MDAwZlx1MDAxNVx1MDAxMG1cdTAwMThcdTAwMTFYoJuL1UTQNVx1MDAxOG+jbaVEXCKNylwiMpac1TpcdTAwMGa2U/H4TqlROMtcdTAwMWXfXHKua5Vibu1ZjVxmb7nhXHUwMDE2fOexXHUwMDFlNOBcdTAwMDQ7nFOqYJ5hmlx1MDAwNWKnhZKadDTlRqFiXGa1ZVE9elx1MDAxZCFgXHUwMDE2XHUwMDE02H+UXHUwMDAwx5qJUjBJXHUwMDE0jpHMqpT54rRPx2nGkcBTSsHEXG5ulSFhrTtxLLyFXHUwMDAydVxi/3CD2plcdTAwMTREYYZcdTAwMDRcdTAwMGaMMY7lM0ZOXHUwMDFjXHUwMDEw5kZlLLOWXHLVWrPF+Fj+XHUwMDAx3zdcblx1MDAxY0BcdTAwMDUh4VhcdTAwMDAqsVx1MDAwMJZTXHUwMDA2s7JUzLJZx4tcdTAwMDXEcHzAuVx1MDAxY/vgolx1MDAxZkbAQ7RcdTAwMTE7T77X8YvvXGJcdTAwMDLxtTJMRSqqXHUwMDE4f4NwPZVq16h39qj6R/1Sakdr6tc+XHUwMDAxR1x1MDAxYZjiKFx1MDAwNP06mKtcdTAwMWRcdTAwMDaghjm4k5CB2Vx1MDAxNDd7X1pcblxmaFx1MDAwZlwiXFwwSWZcdTAwMDVRkTkwYoBFYIRcdTAwMWHGqoSYzIFpMuz6XHUwMDFm09TniyU/MUtSifFcIpaaYp5cdTAwMGK3p1x1MDAxY2M1YZWhjFx1MDAxYVx1MDAwMU4h1pzOZLVYk1x1MDAxYp7OkVx1MDAxMFoo4DqITCWVZFF5MEFovKOGXHUwMDFi3YayL1x1MDAwMVx1MDAxMlx1MDAxMslJaNvmemIvnyml835e3NpcdTAwMTOZXHUwMDFkxFVcdTAwMDGsXHUwMDEzXHTBTIfbW4IzXHUwMDFl3Fx1MDAxYWDoqVFcbuGnokPBkmbzKaOWTEIweaGD/+Wq/X4kpFx1MDAxY0NQgWOsZkgzYSknhKZcdTAwMTBzXHUwMDAygCXGn8zMdIQ+XHUwMDFip8U2bTJcIja3L4dcdTAwMTU1hEY6VjI5p03vPb50uedo+G+Te2LHkUD7mHBSXHJcdTAwMWNnZ9SrxMj5tFtxgk+jnMBcYqI4bYLCXHUwMDA0mFx0tXF+1MqkJkq8a1x1MDAwM6ekm3hP3edgI9hcdTAwMGJcdGYkxrFcdTAwMTBHSIoqu8ldsXloS0ljgomptzZvmpdcdTAwMWI/vmlSjMnhazPAilx1MDAxM3Pd4rbQnrJPhlCaYDxcdTAwMWSZ5FfJXHUwMDE56Wg7Xc9cctqFlrvfud469WrnXHUwMDE3V+9cbvVWKFx1MDAxZtKUTdWSWofDzK9cdTAwMDTHllxuYpwo10c+RI1iXHUwMDFjOzTMXHUwMDEyk1bdzXo/u3Pnui99k8psXqX9w/Lvq5lcdTAwMGLekInSfMGw1CSyX1x1MDAxMtPJ7b55fPTSeaZnnr0u03z78Fjfp+7X3e6Jmiom5Y7AzsHKKmxcdTAwMTY+X8XYMu1+mN6QcVx1MDAxZFBHZn/6/KiO+dX5y8FB6rLQffF66Uo1NLf9WbgwKr7GXHUwMDE43Fx1MDAxY43PJFxuXHUwMDE2Jjks/Ky62L9pK/b0XFw6uHts14+ELa45LJQ1U9WkypFcZr13iCi0tcuJulx1MDAxN1x1MDAwMlx1MDAwYsx12+DupFx1MDAxZjtcdTAwMWK8nvFzb3lcdTAwMTGfSTexalJcYo2JMZpE5rBscjjVypuPtXa+dydvn1xu29WUX8zerXtcIlx1MDAxZK48JCZcckd7XG53hTFCgI1cdTAwMTMujFxcWlx1MDAxMmtcdTAwMDFiUlxi2FxyZcE+o8tcdTAwMDbUV1x1MDAwZetLTfqmLHpsxklcdTAwMTlcIqzh0VxmtJWcgZ5b7vnVI9v2rrJH5Uzj+unh6jS97lxmRMlUMSl1iFDoXHUwMDAyw4RLxXJqXHUwMDA1XHUwMDE3IyZlwJHgeVx1MDAwNHcg+6Kgjd+Dgn5cdTAwMWYxKZhorJhUgseqiIjcII+l3iAm9ZUs7ptcXLeTrpYzle2b9O6D+1x1MDAxOXhoiphUY/EmPDHwXHUwMDE2TYCk1k5Linsxoyb4i4N+N1x1MDAwZfpSksZSmo7NIVx1MDAwModara2OXHUwMDE2yG+/IVlyKd3atti+2G719H2LuvnU2cW6c1x1MDAxYdHTpKTagedcdTAwMGXerlFCcLZcdTAwMWOZ1EKkpENGQGXdXHUwMDE3qf1upPYlJVx1MDAxZOPIJUlJwe2LlZJyXFxcXGTRXHUwMDFiiLOd5Fx1MDAxNPl0UMv2elx1MDAwN437XHUwMDE2y5CrXHUwMDFiM0i79PRcdTAwMTNQ5Fx1MDAxNCUpXHUwMDA3eyGWSY2bQdF1XHUwMDE2cVx0XHUwMDA1k7VVZnXLi19cdTAwMWP5JSTFv97AQfFcckJxIVx1MDAwNbzMaCFpOjlcdTAwMDft9O5Km42OsMXHfK63l8pRL//yXHQ4aJqQXHUwMDE0+1x1MDAwNoErXHUwMDBl6DeCz9mFfclKUk6tXHUwMDAw95/Oqqf+YqFPx0JfStJoJSklXCJ+pV5qsCdKo12r3eS0Nr2hz7pqSYUhjozb2k4qtcBde+OkpFx1MDAxMEFYjUuDhHDw4W2gYvKV14xcdTAwMGW1oFx1MDAxMlx1MDAxM/EnXHUwMDFi7k+j+Vx1MDAwN+1cdTAwMDU6XHUwMDEyly5zL1Da3lLVou9m24/X5Szr9C+KfNTQcyMoLmUwXHUwMDBmXHUwMDE4ayRcdTAwMWZmXHUwMDFhiVx0NPT9KS5cdTAwMTVOqMkmM8HjLGZj0Om90MLjxf61mlx1MDAwMZtcdTAwMTCGe1JNXGaXvml4S1x1MDAxMbguj7g34zGAr1x0658kxtGY36B2nUaYsVpcZuxcdTAwMDdqQ9n3XHUwMDAwX2aS8+V0ycq68qXmXHUwMDFmzpdcdTAwMTH5OTOptrcotYBJNDoh95tcdTAwMTCiv2sz2atW6cFmhE8z2cNzmmpGXHUwMDExXGZcdTAwMDCMSm40sYJqKVx1MDAxNPDMR1x1MDAxMOJ0OV+YXHUwMDEwmVCaMY67ZMCIuZpcdTAwMTivXHUwMDBl6eh0cHO05TDkNKd22eL/UFx1MDAwZc4sSu5cdTAwMWbXa17FS9GoQVEgsZGNQtlecv7jeP9b2aPDR1u7fTxPX+k+V6vjv3dcdTAwMDbCklx0h1xiKrCzXHUwMDBm54Fccip+SEHM4rZCju81XHUwMDBmLFx1MDAwYkHsj/QwUHDEnkpqNOpfa664fUBsv9w11SaPRlx1MDAxNWhcdTAwMWGvQu++dolcdTAwMWZcdTAwMWT3V5f4XCLuJd2AcW1cdTAwMWO2yq6/MWzJ/r35s037xlOjV9/o1vvV6vBtsDevu/EtdJTXzvC+V1xy42QxjePnXHUwMDFiYeJu8kxGksDPsK3QvtmBgdDKVfF698XffCh2XHUwMDAyOopopkAoQPDr4Fx1MDAxYS88Ulx1MDAwZeFjeFx1MDAxYiVusL26VoZBwFx1MDAwYsYqJ6zYXHUwMDE4mG64htDTUNysJcKKIf6GmFx1MDAxYn6vtKC4QetcdTAwMDQ44SRSWkuohMNcdTAwMTg7RXtVJaZMyHTAVate2b6rs2GsunyezdjfXHUwMDA00Fx1MDAwMFx1MDAwN1x1MDAxZn2bXHLbX16S1Ebi/rJcdTAwMTN0j1x1MDAxNzWltyRTXHUwMDE0oitcdTAwMWTgnni6f5OQu95cdTAwMTiYzfQtTVx1MDAxZlxc3KRa5499y9PB3Vx1MDAwZie899dPplx1MDAwYrk5VVx1MDAwNFxiXHUwMDEzXHUwMDE36N5DQL84peqjbW60O61eq9zyXHUwMDAztlx1MDAxMmAo5lxiXCJcdDAzXHUwMDA334WwgPFPYazlb1NcdTAwMTE79sVcdTAwMTCJ8Mjp7YtSV7nTk1xc4/Z4z4iT2yREXHUwMDAyXHUwMDE4d8hQj6FcdTAwMTSHuD1EJCyA+tfiXHUwMDA27lx1MDAxOJxcdTAwMDAlobhJuo3QIFx06ZDQXHUwMDE2XHUwMDFhU+b19+Wof1x1MDAwYqrIvYEqUG1cZnROJtZH8KJYrGBcdTAwMGbpXHUwMDFku9xcdTAwMDTWVZJTRWggS4Bz11x1MDAwM6/db3S9yvem36ohzL43277bjFx1MDAwNjZcckW3wZaXXHUwMDFmXG7sXHUwMDA0V7FcdTAwMTiI90+pUVtccqEpfdlKXHJ80qK7NFx0xCnA2kH/z2JATIRcdTAwMGXnXHKAJ51JXHUwMDBmQcDtVlx1MDAwMsCrXHSF30TJcuzYXHUwMDAzXHUwMDE5fWc2zD2C+339XHUwMDExMD9+XHUwMDAzzFx1MDAxMa1SiMn8XHUwMDE3XlQg6JsohFx1MDAwNU9Nalx1MDAxYWDaN3hcdTAwMDTxc7bRxNj3XHUwMDA1XHK/QF5yfVx1MDAwMEfPXHUwMDA1TODU96mgXHUwMDFkO/Z5XHUwMDAxPStcdTAwMDPAbeyjxlwiXGIjbEDQXHUwMDE4MLWRrza75Nmki7eVg1x1MDAxM3tQ6Z6fvOw1r07T1+ueXHUwMDAwgFjEgWBcdTAwMTbzTUPJYqDg7EfcXHUwMDEzv/adJFx1MDAwNCmxKitFrH1TgrFcdTAwMTCQoVx1MDAxNFhcbmdcIpaIxGTMT4xcdTAwMDI/nX9QXG50zjn6XTH/7lx1MDAwZlx1MDAxZreQP/zePNhJbW/sdFx1MDAxYWC+f682rp89imXDV8jYTlx1MDAxZFxmXHUwMDBiOrDlVyB2XHUwMDFi4TefXHUwMDFjv5u93FV2a3vQPPf28neD48vCibvCXHUwMDA1jHfiV07Fr5jSnmNe/FKOMY40ylJBXCJiXHUwMDE2Klx1MDAxZEs5J1RYRbRcYno8ry2ErJZcbpXwn1x1MDAxZdE/ymVcdTAwMDXueGnAXHUwMDFlXHUwMDAzRVwiM1x1MDAxMP73XHUwMDA2/MPrXHUwMDAw2jb8Rqnjdp5Xi+1p5182qlVcdTAwMDCz4/6XxepTmFx1MDAxYlwiZ+XT5Ki+fC52XG531M3c7F083r5kd04k5euOamWII0Y58bGsPIXYQ3AuLJNgbvNt9lx1MDAxMVx1MDAwN3BwXHUwMDBiXHUwMDE07jUsuNFcZlwilqidq8fxXHUwMDFkuD+/NoDl2C/XqOhOJH9cdTAwMDa+/95otXtg2a6/YmBHnfjNiP7rZ1x1MDAwMvKb225cdTAwMTd6cGe//Vpccv322PCeUtF6R3hhXCJzyFx1MDAwN9+GeX782X/++s//XHUwMDAzJjxA9yJ9 + + + + + DistributedQueryPlannerExecution GraphsMetricsgRPC ServiceFlight SQL API**REST API**Prometheus**gRPC ServiceSessionContextgRPC ServiceQuery StagesBallista SchedulerBallista ExecutorDatafusion Execution (Task Context)ShuffleReadShuffleWriteObject StorageOrFile SystemTransient Local File Storage with shuffle files flight protocolserialisedlogicalplanballistaprotoFlight SQLJDBC Driver** external library** optional \ No newline at end of file diff --git a/docs/source/user-guide/configs.md b/docs/source/user-guide/configs.md index 880ee4901..287b5fe02 100644 --- a/docs/source/user-guide/configs.md +++ b/docs/source/user-guide/configs.md @@ -19,46 +19,74 @@ # Configuration -## BallistaContext Configuration Settings +## Ballista Configuration Settings -Ballista has a number of configuration settings that can be specified when creating a BallistaContext. +Configuring Ballista is quite similar to configuring DataFusion. Most settings are identical, with only a few configurations specific to Ballista. _Example: Specifying configuration options when creating a context_ ```rust -let config = BallistaConfig::builder() -.set("ballista.shuffle.partitions", "200") -.set("ballista.batch.size", "16384") -.build() ?; +use ballista::extension::{SessionConfigExt, SessionContextExt}; -let ctx = BallistaContext::remote("localhost", 50050, & config).await?; +let session_config = SessionConfig::new_with_ballista() + .with_information_schema(true) + .with_ballista_job_name("Super Cool Ballista App"); + +let state = SessionStateBuilder::new() + .with_default_features() + .with_config(session_config) + .build(); + +let ctx: SessionContext = SessionContext::remote_with_state(&url,state).await?; +``` + +`SessionConfig::new_with_ballista()` will setup `SessionConfig` for use with ballista. This is not required, `SessionConfig::new` could be used, but it's advised as it will set up some sensible configuration defaults . + +`SessionConfigExt` expose set of `SessionConfigExt::with_ballista_` and `SessionConfigExt::ballista_` methods which can tune retrieve ballista specific options. + +Notable `SessionConfigExt` configuration methods would be: + +```rust +/// Overrides ballista's [LogicalExtensionCodec] +fn with_ballista_logical_extension_codec( + self, + codec: Arc, +) -> SessionConfig; + +/// Overrides ballista's [PhysicalExtensionCodec] +fn with_ballista_physical_extension_codec( + self, + codec: Arc, +) -> SessionConfig; + +/// Overrides ballista's [QueryPlanner] +fn with_ballista_query_planner( + self, + planner: Arc, +) -> SessionConfig; ``` -### Ballista Configuration Settings - -| key | type | default | description | -| --------------------------------- | ------- | ------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -| ballista.job.name | Utf8 | N/A | Sets the job name that will appear in the web user interface for any submitted jobs. | -| ballista.shuffle.partitions | UInt16 | 16 | Sets the default number of partitions to create when repartitioning query stages. | -| ballista.batch.size | UInt16 | 8192 | Sets the default batch size. | -| ballista.repartition.joins | Boolean | true | When set to true, Ballista will repartition data using the join keys to execute joins in parallel using the provided `ballista.shuffle.partitions` level. | -| ballista.repartition.aggregations | Boolean | true | When set to true, Ballista will repartition data using the aggregate keys to execute aggregates in parallel using the provided `ballista.shuffle.partitions` level. | -| ballista.repartition.windows | Boolean | true | When set to true, Ballista will repartition data using the partition keys to execute window functions in parallel using the provided `ballista.shuffle.partitions` level. | -| ballista.parquet.pruning | Boolean | true | Determines whether Parquet pruning should be enabled or not. | -| ballista.with_information_schema | Boolean | true | Determines whether the `information_schema` should be created in the context. This is necessary for supporting DDL commands such as `SHOW TABLES`. | - -### DataFusion Configuration Settings - -In addition to Ballista-specific configuration settings, the following DataFusion settings can also be specified. - -| key | type | default | description | -| ----------------------------------------------- | ------- | ------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -| datafusion.execution.coalesce_batches | Boolean | true | When set to true, record batches will be examined between each operator and small batches will be coalesced into larger batches. This is helpful when there are highly selective filters or joins that could produce tiny output batches. The target batch size is determined by the configuration setting 'datafusion.execution.coalesce_target_batch_size'. | -| datafusion.execution.coalesce_target_batch_size | UInt64 | 4096 | Target batch size when coalescing batches. Uses in conjunction with the configuration setting 'datafusion.execution.coalesce_batches'. | -| datafusion.explain.logical_plan_only | Boolean | false | When set to true, the explain statement will only print logical plans. | -| datafusion.explain.physical_plan_only | Boolean | false | When set to true, the explain statement will only print physical plans. | -| datafusion.optimizer.filter_null_join_keys | Boolean | false | When set to true, the optimizer will insert filters before a join between a nullable and non-nullable column to filter out nulls on the nullable side. This filter can add additional overhead when the file format does not fully support predicate push down. | -| datafusion.optimizer.skip_failed_rules | Boolean | true | When set to true, the logical plan optimizer will produce warning messages if any optimization rules produce errors and then proceed to the next rule. When set to false, any rules that produce errors will cause the query to fail. | +which could be used to change default ballista behavior. + +If information schema is enabled all configuration parameters could be retrieved or set using SQL; + +```rust +let ctx: SessionContext = SessionContext::remote_with_state(&url, state).await?; + +let result = ctx + .sql("select name, value from information_schema.df_settings where name like 'ballista'") + .await? + .collect() + .await?; + +let expected = [ + "+-------------------+-------------------------+", + "| name | value |", + "+-------------------+-------------------------+", + "| ballista.job.name | Super Cool Ballista App |", + "+-------------------+-------------------------+", +]; +``` ## Ballista Scheduler Configuration Settings diff --git a/docs/source/user-guide/deployment/quick-start.md b/docs/source/user-guide/deployment/quick-start.md index d94de0475..df8434beb 100644 --- a/docs/source/user-guide/deployment/quick-start.md +++ b/docs/source/user-guide/deployment/quick-start.md @@ -19,16 +19,14 @@ # Ballista Quickstart -A simple way to start a local cluster for testing purposes is to use cargo to build the project and then run the scheduler and executor binaries directly along with the Ballista UI. +A simple way to start a local cluster for testing purposes is to use cargo to build the project and then run the scheduler and executor binaries directly. Project Requirements: - [Rust](https://www.rust-lang.org/tools/install) - [Protobuf Compiler](https://protobuf.dev/downloads/) -- [Node.js](https://nodejs.org/en/download) -- [Yarn](https://classic.yarnpkg.com/lang/en/docs/install) -### Build the project +## Build the project From the root of the project, build release binaries. @@ -55,39 +53,47 @@ RUST_LOG=info ./target/release/ballista-executor -c 2 -p 50052 The examples can be run using the `cargo run --bin` syntax. Open a new terminal session and run the following commands. -## Running the examples - -## Distributed SQL Example +### Distributed SQL Example ```bash cd examples cargo run --release --example remote-sql ``` -### Source code for distributed SQL example +#### Source code for distributed SQL example ```rust use ballista::prelude::*; -use datafusion::prelude::CsvReadOptions; +use ballista_examples::test_util; +use datafusion::{ + execution::SessionStateBuilder, + prelude::{CsvReadOptions, SessionConfig, SessionContext}, +}; /// This example demonstrates executing a simple query against an Arrow data source (CSV) and /// fetching results, using SQL #[tokio::main] async fn main() -> Result<()> { - let config = BallistaConfig::builder() - .set("ballista.shuffle.partitions", "4") - .build()?; - let ctx = BallistaContext::remote("localhost", 50050, &config).await?; + let config = SessionConfig::new_with_ballista() + .with_target_partitions(4) + .with_ballista_job_name("Remote SQL Example"); + + let state = SessionStateBuilder::new() + .with_config(config) + .with_default_features() + .build(); + + let ctx = SessionContext::remote_with_state("df://localhost:50050", state).await?; + + let test_data = test_util::examples_test_data(); - // register csv file with the execution context ctx.register_csv( "test", - "testdata/aggregate_test_100.csv", + &format!("{test_data}/aggregate_test_100.csv"), CsvReadOptions::new(), ) .await?; - // execute the query let df = ctx .sql( "SELECT c1, MIN(c12), MAX(c12) \ @@ -97,40 +103,42 @@ async fn main() -> Result<()> { ) .await?; - // print the results df.show().await?; Ok(()) } ``` -## Distributed DataFrame Example +### Distributed DataFrame Example ```bash cd examples cargo run --release --example remote-dataframe ``` -### Source code for distributed DataFrame example +#### Source code for distributed DataFrame example ```rust +use ballista::prelude::*; +use ballista_examples::test_util; +use datafusion::{ + prelude::{col, lit, ParquetReadOptions, SessionContext}, +}; + #[tokio::main] async fn main() -> Result<()> { - let config = BallistaConfig::builder() - .set("ballista.shuffle.partitions", "4") - .build()?; - let ctx = BallistaContext::remote("localhost", 50050, &config).await?; + // creating SessionContext with default settings + let ctx = SessionContext::remote("df://localhost:50050".await?; - let filename = "testdata/alltypes_plain.parquet"; + let test_data = test_util::examples_test_data(); + let filename = format!("{test_data}/alltypes_plain.parquet"); - // define the query using the DataFrame trait let df = ctx .read_parquet(filename, ParquetReadOptions::default()) .await? .select_columns(&["id", "bool_col", "timestamp_col"])? .filter(col("id").gt(lit(1)))?; - // print the results df.show().await?; Ok(()) diff --git a/docs/source/user-guide/faq.md b/docs/source/user-guide/faq.md index 5cfd9fe8b..817bd6c54 100644 --- a/docs/source/user-guide/faq.md +++ b/docs/source/user-guide/faq.md @@ -25,4 +25,4 @@ DataFusion is a library for executing queries in-process using the Apache Arrow model and computational kernels. It is designed to run within a single process, using threads for parallel query execution. -Ballista is a distributed compute platform built on DataFusion. +Ballista is a distributed compute platform for DataFusion workloads. diff --git a/docs/source/user-guide/flightsql.md b/docs/source/user-guide/flightsql.md index 4572eef91..b6bb71f4a 100644 --- a/docs/source/user-guide/flightsql.md +++ b/docs/source/user-guide/flightsql.md @@ -21,6 +21,8 @@ One of the easiest ways to start with Ballista is to plug it into your existing data infrastructure using support for Arrow Flight SQL JDBC. +> This is optional scheduler feature which should be enabled with `flight-sql` feature + Getting started involves these main steps: 1. [Installing prerequisites](#prereq) diff --git a/docs/source/user-guide/metrics.md b/docs/source/user-guide/metrics.md index 7e831dcab..c061f3d44 100644 --- a/docs/source/user-guide/metrics.md +++ b/docs/source/user-guide/metrics.md @@ -21,6 +21,8 @@ ## Prometheus +> This is optional scheduler feature which should be enabled with `prometheus-metrics` feature + Built with default features, the ballista scheduler will automatically collect and expose a standard set of prometheus metrics. The metrics currently collected automatically include: diff --git a/docs/source/user-guide/rust.md b/docs/source/user-guide/rust.md index e3015e763..04e412b8b 100644 --- a/docs/source/user-guide/rust.md +++ b/docs/source/user-guide/rust.md @@ -17,78 +17,126 @@ under the License. --> -# Ballista Rust Client +# Distributing DataFusion with Ballista -To connect to a Ballista cluster from Rust, first start by creating a `BallistaContext`. +To connect to a Ballista cluster from Rust, first start by creating a `SessionContext` connected to remote scheduler server. ```rust -let config = BallistaConfig::builder() - .set("ballista.shuffle.partitions", "4") - .build()?; +use ballista::prelude::*; +use datafusion::{ + execution::SessionStateBuilder, + prelude::{SessionConfig, SessionContext}, +}; + +let config = SessionConfig::new_with_ballista() + .with_target_partitions(4) + .with_ballista_job_name("Remote SQL Example"); + +let state = SessionStateBuilder::new() + .with_config(config) + .with_default_features() + .build(); + +let ctx = SessionContext::remote_with_state("df://localhost:50050", state).await?; +``` + +For testing purposes, standalone, in process cluster could be started with: + +```rust +use ballista::prelude::*; +use datafusion::{ + execution::SessionStateBuilder, + prelude::{SessionConfig, SessionContext}, +}; +let config = SessionConfig::new_with_ballista() + .with_target_partitions(1) + .with_ballista_standalone_parallelism(2); + +let state = SessionStateBuilder::new() + .with_config(config) + .with_default_features() + .build(); + +let ctx = SessionContext::standalone_with_state(state).await?; -// connect to Ballista scheduler -let ctx = BallistaContext::remote("localhost", 50050, &config); ``` -Here is a full example using the DataFrame API. +Following examples require running remove scheduler and executor nodes. + +Full example using the DataFrame API. ```rust +use ballista::prelude::*; +use ballista_examples::test_util; +use datafusion::{ + prelude::{col, lit, ParquetReadOptions, SessionContext}, +}; + +/// This example demonstrates executing a simple query against an Arrow data source (Parquet) and +/// fetching results, using the DataFrame trait #[tokio::main] async fn main() -> Result<()> { - let config = BallistaConfig::builder() - .set("ballista.shuffle.partitions", "4") - .build()?; + // creating SessionContext with default settings + let ctx = SessionContext::remote("df://localhost:50050").await?; - // connect to Ballista scheduler - let ctx = BallistaContext::remote("localhost", 50050, &config); + let test_data = test_util::examples_test_data(); + let filename = format!("{test_data}/alltypes_plain.parquet"); - let testdata = datafusion::test_util::parquet_test_data(); - - let filename = &format!("{}/alltypes_plain.parquet", testdata); - - // define the query using the DataFrame trait let df = ctx - .read_parquet(filename)? + .read_parquet(filename, ParquetReadOptions::default()) + .await? .select_columns(&["id", "bool_col", "timestamp_col"])? .filter(col("id").gt(lit(1)))?; - // print the results df.show().await?; Ok(()) } ``` -Here is a full example demonstrating SQL usage. +Here is a full example demonstrating SQL usage, with user specific `SessionConfig`: ```rust +use ballista::prelude::*; +use ballista_examples::test_util; +use datafusion::{ + execution::SessionStateBuilder, + prelude::{CsvReadOptions, SessionConfig, SessionContext}, +}; + +/// This example demonstrates executing a simple query against an Arrow data source (CSV) and +/// fetching results, using SQL #[tokio::main] async fn main() -> Result<()> { - let config = BallistaConfig::builder() - .set("ballista.shuffle.partitions", "4") - .build()?; + let config = SessionConfig::new_with_ballista() + .with_target_partitions(4) + .with_ballista_job_name("Remote SQL Example"); - // connect to Ballista scheduler - let ctx = BallistaContext::remote("localhost", 50050, &config); + let state = SessionStateBuilder::new() + .with_config(config) + .with_default_features() + .build(); - let testdata = datafusion::test_util::arrow_test_data(); + let ctx = SessionContext::remote_with_state("df://localhost:50050", state).await?; + + let test_data = test_util::examples_test_data(); - // register csv file with the execution context ctx.register_csv( - "aggregate_test_100", - &format!("{}/csv/aggregate_test_100.csv", testdata), + "test", + &format!("{test_data}/aggregate_test_100.csv"), CsvReadOptions::new(), - )?; + ) + .await?; - // execute the query - let df = ctx.sql( - "SELECT c1, MIN(c12), MAX(c12) \ - FROM aggregate_test_100 \ + let df = ctx + .sql( + "SELECT c1, MIN(c12), MAX(c12) \ + FROM test \ WHERE c11 > 0.1 AND c11 < 0.9 \ GROUP BY c1", - )?; + ) + .await?; - // print the results df.show().await?; Ok(()) diff --git a/docs/source/user-guide/scheduler.md b/docs/source/user-guide/scheduler.md index f6a3ca6a8..80bc752e8 100644 --- a/docs/source/user-guide/scheduler.md +++ b/docs/source/user-guide/scheduler.md @@ -23,6 +23,8 @@ The scheduler also provides a REST API that allows jobs to be monitored. +> This is optional scheduler feature which should be enabled with `rest-api` feature + | API | Method | Description | | --------------------- | ------ | ----------------------------------------------------------- | | /api/jobs | GET | Get a list of jobs that have been submitted to the cluster. | diff --git a/examples/README.md b/examples/README.md index 4eaaac1e2..14604ac2b 100644 --- a/examples/README.md +++ b/examples/README.md @@ -21,7 +21,7 @@ This directory contains examples for executing distributed queries with Ballista. -# Standalone Examples +## Standalone Examples The standalone example is the easiest to get started with. Ballista supports a standalone mode where a scheduler and executor are started in-process. @@ -33,18 +33,35 @@ cargo run --example standalone_sql --features="ballista/standalone" ### Source code for standalone SQL example ```rust +use ballista::{ + extension::SessionConfigExt, + prelude::* +}; +use datafusion::{ + execution::{options::ParquetReadOptions, SessionStateBuilder}, + prelude::{SessionConfig, SessionContext}, +}; + #[tokio::main] async fn main() -> Result<()> { - let config = BallistaConfig::builder() - .set("ballista.shuffle.partitions", "1") - .build()?; + let config = SessionConfig::new_with_ballista() + .with_target_partitions(1) + .with_ballista_standalone_parallelism(2); - let ctx = BallistaContext::standalone(&config, 2).await?; + let state = SessionStateBuilder::new() + .with_config(config) + .with_default_features() + .build(); - ctx.register_csv( + let ctx = SessionContext::standalone_with_state(state).await?; + + let test_data = test_util::examples_test_data(); + + // register parquet file with the execution context + ctx.register_parquet( "test", - "testdata/aggregate_test_100.csv", - CsvReadOptions::new(), + &format!("{test_data}/alltypes_plain.parquet"), + ParquetReadOptions::default(), ) .await?; @@ -56,12 +73,12 @@ async fn main() -> Result<()> { ``` -# Distributed Examples +## Distributed Examples For background information on the Ballista architecture, refer to the [Ballista README](../ballista/client/README.md). -## Start a standalone cluster +### Start a standalone cluster From the root of the project, build release binaries. @@ -83,40 +100,49 @@ RUST_LOG=info ./target/release/ballista-executor -c 2 -p 50051 RUST_LOG=info ./target/release/ballista-executor -c 2 -p 50052 ``` -## Running the examples +### Running the examples The examples can be run using the `cargo run --bin` syntax. -## Distributed SQL Example +### Distributed SQL Example ```bash cargo run --release --example remote-sql ``` -### Source code for distributed SQL example +#### Source code for distributed SQL example ```rust -use ballista::prelude::*; -use datafusion::prelude::CsvReadOptions; +use ballista::{extension::SessionConfigExt, prelude::*}; +use datafusion::{ + execution::SessionStateBuilder, + prelude::{CsvReadOptions, SessionConfig, SessionContext}, +}; /// This example demonstrates executing a simple query against an Arrow data source (CSV) and /// fetching results, using SQL #[tokio::main] async fn main() -> Result<()> { - let config = BallistaConfig::builder() - .set("ballista.shuffle.partitions", "4") - .build()?; - let ctx = BallistaContext::remote("localhost", 50050, &config).await?; + let config = SessionConfig::new_with_ballista() + .with_target_partitions(4) + .with_ballista_job_name("Remote SQL Example"); + + let state = SessionStateBuilder::new() + .with_config(config) + .with_default_features() + .build(); + + let ctx = SessionContext::remote_with_state("df://localhost:50050", state).await?; + + let test_data = test_util::examples_test_data(); - // register csv file with the execution context ctx.register_csv( "test", - "testdata/aggregate_test_100.csv", + &format!("{test_data}/aggregate_test_100.csv"), CsvReadOptions::new(), ) .await?; - // execute the query let df = ctx .sql( "SELECT c1, MIN(c12), MAX(c12) \ @@ -126,39 +152,49 @@ async fn main() -> Result<()> { ) .await?; - // print the results df.show().await?; Ok(()) } ``` -## Distributed DataFrame Example +### Distributed DataFrame Example ```bash cargo run --release --example remote-dataframe ``` -### Source code for distributed DataFrame example +#### Source code for distributed DataFrame example ```rust +use ballista::{extension::SessionConfigExt, prelude::*}; +use datafusion::{ + execution::SessionStateBuilder, + prelude::{col, lit, ParquetReadOptions, SessionConfig, SessionContext}, +}; + +/// This example demonstrates executing a simple query against an Arrow data source (Parquet) and +/// fetching results, using the DataFrame trait #[tokio::main] async fn main() -> Result<()> { - let config = BallistaConfig::builder() - .set("ballista.shuffle.partitions", "4") - .build()?; - let ctx = BallistaContext::remote("localhost", 50050, &config).await?; + let config = SessionConfig::new_with_ballista().with_target_partitions(4); + + let state = SessionStateBuilder::new() + .with_config(config) + .with_default_features() + .build(); + + let ctx = SessionContext::remote_with_state("df://localhost:50050", state).await?; - let filename = "testdata/alltypes_plain.parquet"; + let test_data = test_util::examples_test_data(); + let filename = format!("{test_data}/alltypes_plain.parquet"); - // define the query using the DataFrame trait let df = ctx .read_parquet(filename, ParquetReadOptions::default()) .await? .select_columns(&["id", "bool_col", "timestamp_col"])? .filter(col("id").gt(lit(1)))?; - // print the results df.show().await?; Ok(())