| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | use super::schema::{instrumentations, work_parts, work_sections, works};
 | 
					
						
							| 
									
										
										
										
											2020-11-29 01:29:03 +01:00
										 |  |  | use super::{get_instrument, get_person, update_instrument, update_person};
 | 
					
						
							|  |  |  | use super::{DbConn, Instrument, Person, User};
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | use crate::error::ServerError;
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | use anyhow::{anyhow, Error, Result};
 | 
					
						
							|  |  |  | use diesel::prelude::*;
 | 
					
						
							|  |  |  | use serde::{Deserialize, Serialize};
 | 
					
						
							|  |  |  | use std::convert::TryInto;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | /// A specific work by a composer.
 | 
					
						
							|  |  |  | #[derive(Serialize, Deserialize, Debug, Clone)]
 | 
					
						
							|  |  |  | #[serde(rename_all = "camelCase")]
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | pub struct Work {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |     pub id: String,
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     pub title: String,
 | 
					
						
							|  |  |  |     pub composer: Person,
 | 
					
						
							|  |  |  |     pub instruments: Vec<Instrument>,
 | 
					
						
							|  |  |  |     pub parts: Vec<WorkPart>,
 | 
					
						
							|  |  |  |     pub sections: Vec<WorkSection>,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// A playable part of a work.
 | 
					
						
							|  |  |  | #[derive(Serialize, Deserialize, Debug, Clone)]
 | 
					
						
							|  |  |  | #[serde(rename_all = "camelCase")]
 | 
					
						
							|  |  |  | pub struct WorkPart {
 | 
					
						
							|  |  |  |     pub title: String,
 | 
					
						
							|  |  |  |     pub composer: Option<Person>,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// A heading within the work structure.
 | 
					
						
							|  |  |  | #[derive(Serialize, Deserialize, Debug, Clone)]
 | 
					
						
							|  |  |  | #[serde(rename_all = "camelCase")]
 | 
					
						
							|  |  |  | pub struct WorkSection {
 | 
					
						
							|  |  |  |     pub title: String,
 | 
					
						
							|  |  |  |     pub before_index: i64,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Table data for a work.
 | 
					
						
							|  |  |  | #[derive(Insertable, Queryable, Debug, Clone)]
 | 
					
						
							|  |  |  | #[table_name = "works"]
 | 
					
						
							|  |  |  | struct WorkRow {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |     pub id: String,
 | 
					
						
							|  |  |  |     pub composer: String,
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     pub title: String,
 | 
					
						
							|  |  |  |     pub created_by: String,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | /// Table data for an instrumentation.
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | #[derive(Insertable, Queryable, Debug, Clone)]
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | #[table_name = "instrumentations"]
 | 
					
						
							|  |  |  | struct InstrumentationRow {
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     pub id: i64,
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |     pub work: String,
 | 
					
						
							|  |  |  |     pub instrument: String,
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | /// Table data for a work part.
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | #[derive(Insertable, Queryable, Debug, Clone)]
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | #[table_name = "work_parts"]
 | 
					
						
							|  |  |  | struct WorkPartRow {
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     pub id: i64,
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |     pub work: String,
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     pub part_index: i64,
 | 
					
						
							|  |  |  |     pub title: String,
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |     pub composer: Option<String>,
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | /// Table data for a work section.
 | 
					
						
							|  |  |  | #[table_name = "work_sections"]
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | #[derive(Insertable, Queryable, Debug, Clone)]
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | struct WorkSectionRow {
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     pub id: i64,
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |     pub work: String,
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     pub title: String,
 | 
					
						
							|  |  |  |     pub before_index: i64,
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | /// Update an existing work or insert a new one. This will only succeed, if the user is allowed to
 | 
					
						
							|  |  |  | /// do that.
 | 
					
						
							|  |  |  | pub fn update_work(conn: &DbConn, work: &Work, user: &User) -> Result<()> {
 | 
					
						
							|  |  |  |     conn.transaction::<(), Error, _>(|| {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |         let old_row = get_work_row(conn, &work.id)?;
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |         let allowed = match old_row {
 | 
					
						
							|  |  |  |             Some(row) => user.may_edit(&row.created_by),
 | 
					
						
							|  |  |  |             None => user.may_create(),
 | 
					
						
							|  |  |  |         };
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |         if allowed {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |             let id = &work.id;
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |             // This will also delete rows from associated tables.
 | 
					
						
							|  |  |  |             diesel::delete(works::table)
 | 
					
						
							|  |  |  |                 .filter(works::id.eq(id))
 | 
					
						
							|  |  |  |                 .execute(conn)?;
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-29 01:29:03 +01:00
										 |  |  |             // Add associated items, if they don't already exist.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if get_person(conn, &work.composer.id)?.is_none() {
 | 
					
						
							|  |  |  |                 update_person(conn, &work.composer, &user)?;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for instrument in &work.instruments {
 | 
					
						
							|  |  |  |                 if get_instrument(conn, &instrument.id)?.is_none() {
 | 
					
						
							|  |  |  |                     update_instrument(conn, instrument, &user)?;
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for part in &work.parts {
 | 
					
						
							|  |  |  |                 if let Some(person) = &part.composer {
 | 
					
						
							|  |  |  |                     if get_person(conn, &person.id)?.is_none() {
 | 
					
						
							|  |  |  |                         update_person(conn, person, &user)?;
 | 
					
						
							|  |  |  |                     }
 | 
					
						
							|  |  |  |                 }
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Add the actual work.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |             let row = WorkRow {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |                 id: id.clone(),
 | 
					
						
							|  |  |  |                 composer: work.composer.id.clone(),
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |                 title: work.title.clone(),
 | 
					
						
							|  |  |  |                 created_by: user.username.clone(),
 | 
					
						
							|  |  |  |             };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             diesel::insert_into(works::table)
 | 
					
						
							|  |  |  |                 .values(row)
 | 
					
						
							|  |  |  |                 .execute(conn)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for instrument in &work.instruments {
 | 
					
						
							|  |  |  |                 diesel::insert_into(instrumentations::table)
 | 
					
						
							|  |  |  |                     .values(InstrumentationRow {
 | 
					
						
							|  |  |  |                         id: rand::random(),
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |                         work: id.clone(),
 | 
					
						
							|  |  |  |                         instrument: instrument.id.clone(),
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |                     })
 | 
					
						
							|  |  |  |                     .execute(conn)?;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (index, part) in work.parts.iter().enumerate() {
 | 
					
						
							|  |  |  |                 let row = WorkPartRow {
 | 
					
						
							|  |  |  |                     id: rand::random(),
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |                     work: id.clone(),
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |                     part_index: index.try_into()?,
 | 
					
						
							|  |  |  |                     title: part.title.clone(),
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |                     composer: part.composer.as_ref().map(|person| person.id.clone()),
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |                 };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 diesel::insert_into(work_parts::table)
 | 
					
						
							|  |  |  |                     .values(row)
 | 
					
						
							|  |  |  |                     .execute(conn)?;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for section in &work.sections {
 | 
					
						
							|  |  |  |                 let row = WorkSectionRow {
 | 
					
						
							|  |  |  |                     id: rand::random(),
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |                     work: id.clone(),
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |                     title: section.title.clone(),
 | 
					
						
							|  |  |  |                     before_index: section.before_index,
 | 
					
						
							|  |  |  |                 };
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 diesel::insert_into(work_sections::table)
 | 
					
						
							|  |  |  |                     .values(row)
 | 
					
						
							|  |  |  |                     .execute(conn)?;
 | 
					
						
							|  |  |  |             }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             Ok(())
 | 
					
						
							|  |  |  |         } else {
 | 
					
						
							|  |  |  |             Err(Error::new(ServerError::Forbidden))
 | 
					
						
							|  |  |  |         }
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     })?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Ok(())
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | /// Get an existing work and all available information from related tables.
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  | pub fn get_work(conn: &DbConn, id: &str) -> Result<Option<Work>> {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     let work = match get_work_row(conn, id)? {
 | 
					
						
							|  |  |  |         Some(row) => Some(get_description_for_work_row(conn, &row)?),
 | 
					
						
							|  |  |  |         None => None,
 | 
					
						
							|  |  |  |     };
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     Ok(work)
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | /// Delete an existing work. This will fail if there are still other tables that relate to
 | 
					
						
							|  |  |  | /// this work except for the things that are part of the information on the work itself. Also,
 | 
					
						
							|  |  |  | /// this will only succeed, if the provided user is allowed to delete the work.
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  | pub fn delete_work(conn: &DbConn, id: &str, user: &User) -> Result<()> {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     if user.may_delete() {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |         diesel::delete(works::table.filter(works::id.eq(id))).execute(conn)?;
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |         Ok(())
 | 
					
						
							|  |  |  |     } else {
 | 
					
						
							|  |  |  |         Err(Error::new(ServerError::Forbidden))
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     }
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | }
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | /// Get all existing works by a composer and related information from other tables.
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  | pub fn get_works(conn: &DbConn, composer_id: &str) -> Result<Vec<Work>> {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     let mut works: Vec<Work> = Vec::new();
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     let rows = works::table
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |         .filter(works::composer.eq(composer_id))
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |         .load::<WorkRow>(conn)?;
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     for row in rows {
 | 
					
						
							|  |  |  |         works.push(get_description_for_work_row(conn, &row)?);
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     Ok(works)
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Get an already existing work without related rows from other tables.
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  | fn get_work_row(conn: &DbConn, id: &str) -> Result<Option<WorkRow>> {
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     Ok(works::table
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |         .filter(works::id.eq(id))
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |         .load::<WorkRow>(conn)?
 | 
					
						
							|  |  |  |         .into_iter()
 | 
					
						
							|  |  |  |         .next())
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Retrieve all available information on a work from related tables.
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  | fn get_description_for_work_row(conn: &DbConn, row: &WorkRow) -> Result<Work> {
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     let mut instruments: Vec<Instrument> = Vec::new();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let instrumentations = instrumentations::table
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |         .filter(instrumentations::work.eq(&row.id))
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |         .load::<InstrumentationRow>(conn)?;
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for instrumentation in instrumentations {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |         let id = instrumentation.instrument.clone();
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |         instruments
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |             .push(get_instrument(conn, &id)?.ok_or(anyhow!("No instrument with ID: {}", id))?);
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     let mut parts: Vec<WorkPart> = Vec::new();
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let part_rows = work_parts::table
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |         .filter(work_parts::work.eq(&row.id))
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |         .load::<WorkPartRow>(conn)?;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for part_row in part_rows {
 | 
					
						
							|  |  |  |         parts.push(WorkPart {
 | 
					
						
							|  |  |  |             title: part_row.title,
 | 
					
						
							|  |  |  |             composer: match part_row.composer {
 | 
					
						
							|  |  |  |                 Some(id) => {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |                     Some(get_person(conn, &id)?.ok_or(anyhow!("No person with ID: {}", id))?)
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |                 }
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |                 None => None,
 | 
					
						
							|  |  |  |             },
 | 
					
						
							|  |  |  |         });
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     let mut sections: Vec<WorkSection> = Vec::new();
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     let section_rows = work_sections::table
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |         .filter(work_sections::work.eq(&row.id))
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |         .load::<WorkSectionRow>(conn)?;
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     for section in section_rows {
 | 
					
						
							|  |  |  |         sections.push(WorkSection {
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |             title: section.title,
 | 
					
						
							|  |  |  |             before_index: section.before_index,
 | 
					
						
							|  |  |  |         });
 | 
					
						
							|  |  |  |     }
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |     let id = &row.composer;
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     let composer = get_person(conn, id)?.ok_or(anyhow!("No person with ID: {}", id))?;
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |     Ok(Work {
 | 
					
						
							| 
									
										
										
										
											2020-11-28 20:30:14 +01:00
										 |  |  |         id: row.id.clone(),
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |         composer,
 | 
					
						
							|  |  |  |         title: row.title.clone(),
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |         instruments,
 | 
					
						
							| 
									
										
										
										
											2020-11-28 00:30:12 +01:00
										 |  |  |         parts,
 | 
					
						
							|  |  |  |         sections,
 | 
					
						
							| 
									
										
										
										
											2020-11-14 23:08:37 +01:00
										 |  |  |     })
 | 
					
						
							|  |  |  | }
 |