From 69f1f69d868840a81546004e6de02d1ed2b07237 Mon Sep 17 00:00:00 2001 From: alexeh Date: Tue, 19 Nov 2024 06:42:34 +0100 Subject: [PATCH 1/5] ugly and buggy first approach to upload user data --- admin/datasource.ts | 2 + admin/index.ts | 13 +- api/src/modules/import/import.controller.ts | 32 +++- api/src/modules/import/import.service.ts | 16 ++ .../import/services/excel-parser.interface.ts | 2 + .../import/services/user-data-parser.ts | 140 ++++++++++++++++++ .../modules/import/services/xlsx.parser.ts | 52 +++++++ api/src/modules/users/users.module.ts | 6 +- shared/contracts/users.contract.ts | 15 +- shared/entities/user-project-data.entity.ts | 139 +++++++++++++++++ shared/entities/users/user.entity.ts | 8 + shared/lib/db-entities.ts | 2 + 12 files changed, 420 insertions(+), 7 deletions(-) create mode 100644 api/src/modules/import/services/user-data-parser.ts create mode 100644 shared/entities/user-project-data.entity.ts diff --git a/admin/datasource.ts b/admin/datasource.ts index fa59605c..cbb22fe7 100644 --- a/admin/datasource.ts +++ b/admin/datasource.ts @@ -29,10 +29,12 @@ import { ImplementationLaborCost } from "@shared/entities/cost-inputs/implementa import { BaseSize } from "@shared/entities/base-size.entity.js"; import { BaseIncrease } from "@shared/entities/base-increase.entity.js"; import { ModelAssumptions } from "@shared/entities/model-assumptions.entity.js"; +import { UserUploadedData } from "@shared/entities/user-project-data.entity.js"; // TODO: If we import the COMMON_DATABASE_ENTITIES from shared, we get an error where DataSouce is not set for a given entity export const ADMINJS_ENTITIES = [ User, + UserUploadedData, ApiEventsEntity, Country, ProjectSize, diff --git a/admin/index.ts b/admin/index.ts index 5e0e41a0..fe9ba923 100644 --- a/admin/index.ts +++ b/admin/index.ts @@ -33,6 +33,7 @@ import { ImplementationLaborCostResource } from "./resources/implementation-labo import { BaseSizeResource } from "./resources/base-size/base-size.resource.js"; import { BaseIncreaseResource } from "./resources/base-increase/base-increase.resource.js"; import { ModelAssumptionResource } from "./resources/model-assumptions/model-assumptions.resource.js"; +import { UserUploadedData } from "@shared/entities/user-project-data.entity.js"; AdminJS.registerAdapter({ Database: AdminJSTypeorm.Database, @@ -93,6 +94,16 @@ const start = async () => { icon: "Globe", }, }, + { + resource: UserUploadedData, + name: "UserUploadedData", + options: { + navigation: { + name: "Data Management", + icon: "Database", + }, + }, + }, ], locale: { language: "en", @@ -119,7 +130,7 @@ const start = async () => { app.listen(PORT, () => { console.log( - `AdminJS started on http://localhost:${PORT}${admin.options.rootPath}` + `AdminJS started on http://localhost:${PORT}${admin.options.rootPath}`, ); }); }; diff --git a/api/src/modules/import/import.controller.ts b/api/src/modules/import/import.controller.ts index 9f205a7f..cd0b4d28 100644 --- a/api/src/modules/import/import.controller.ts +++ b/api/src/modules/import/import.controller.ts @@ -1,5 +1,11 @@ -import { Controller, UseGuards, UseInterceptors } from '@nestjs/common'; -import { FileInterceptor } from '@nestjs/platform-express'; +import { + Controller, + HttpStatus, + UploadedFiles, + UseGuards, + UseInterceptors, +} from '@nestjs/common'; +import { FileInterceptor, FilesInterceptor } from '@nestjs/platform-express'; import { JwtAuthGuard } from '@api/modules/auth/guards/jwt-auth.guard'; import { RolesGuard } from '@api/modules/auth/guards/roles.guard'; import { RequiredRoles } from '@api/modules/auth/decorators/roles.decorator'; @@ -13,6 +19,8 @@ import { ControllerResponse } from '@api/types/controller-response.type'; import { Multer } from 'multer'; import { GetUser } from '@api/modules/auth/decorators/get-user.decorator'; import { User } from '@shared/entities/users/user.entity'; +import { usersContract } from '@shared/contracts/users.contract'; +import { Public } from '@api/modules/auth/decorators/is-public.decorator'; @Controller() @UseGuards(JwtAuthGuard, RolesGuard) @@ -36,4 +44,24 @@ export class ImportController { }; }); } + + //@Public() + @UseInterceptors(FilesInterceptor('files', 2)) + @RequiredRoles(ROLES.PARTNER, ROLES.ADMIN) + @TsRestHandler(usersContract.uploadData) + async uploadData( + @GetUser() user: User, + @UploadedFiles() files: Array, + ): Promise { + return tsRestHandler(usersContract.uploadData, async () => { + console.log(files); + const [file1, file2] = files; + const [file1Buffer, file2Buffer] = [file1.buffer, file2.buffer]; + const data = await this.service.importDataProvidedByPartner( + [file1Buffer, file2Buffer], + user.id, + ); + return { body: data, status: HttpStatus.OK }; + }); + } } diff --git a/api/src/modules/import/import.service.ts b/api/src/modules/import/import.service.ts index da4de929..0834dd47 100644 --- a/api/src/modules/import/import.service.ts +++ b/api/src/modules/import/import.service.ts @@ -8,6 +8,12 @@ import { ImportRepository } from '@api/modules/import/import.repostiory'; import { EventBus } from '@nestjs/cqrs'; import { API_EVENT_TYPES } from '@api/modules/api-events/events.enum'; import { ImportEvent } from '@api/modules/import/events/import.event'; +import { DataSource } from 'typeorm'; +import { + userDataInputJson, + userDataMapJsonToEntity, +} from '@api/modules/import/services/user-data-parser'; +import { UserUploadedData } from '@shared/entities/user-project-data.entity'; @Injectable() export class ImportService { @@ -24,6 +30,7 @@ export class ImportService { private readonly importRepo: ImportRepository, private readonly preprocessor: EntityPreprocessor, private readonly eventBus: EventBus, + private readonly dataSource: DataSource, ) {} async import(fileBuffer: Buffer, userId: string) { @@ -44,4 +51,13 @@ export class ImportService { registerImportEvent(userId: string, eventType: typeof this.eventMap) { this.eventBus.publish(new ImportEvent(eventType, userId, {})); } + + async importDataProvidedByPartner(fileBuffers: Buffer[], userId: string) { + const { costInputs } = await this.excelParser.parseUserExcels(fileBuffers); + const mapped = userDataMapJsonToEntity(userDataInputJson, userId); + const savedData = await this.dataSource + .getRepository(UserUploadedData) + .save(mapped); + return savedData; + } } diff --git a/api/src/modules/import/services/excel-parser.interface.ts b/api/src/modules/import/services/excel-parser.interface.ts index 0f9bb1d6..315b8e94 100644 --- a/api/src/modules/import/services/excel-parser.interface.ts +++ b/api/src/modules/import/services/excel-parser.interface.ts @@ -32,4 +32,6 @@ export const SHEETS_TO_PARSE = [ export interface ExcelParserInterface { parseExcel(data: Buffer): Promise; + + parseUserExcels(data: Buffer[]): Promise; } diff --git a/api/src/modules/import/services/user-data-parser.ts b/api/src/modules/import/services/user-data-parser.ts new file mode 100644 index 00000000..a4f109f4 --- /dev/null +++ b/api/src/modules/import/services/user-data-parser.ts @@ -0,0 +1,140 @@ +// I feel dirty doing this... + +import { UserUploadedData } from '@shared/entities/user-project-data.entity'; + +export const userDataInputJson = { + 'Program name (if willing to share)': 'test name', + 'Intended length of project': 2000, + Country: 'Basque Country', + Currency: 'Euro', + 'Project start year': 1, + 'Project activity': 'Conservation', + Ecosystem: 'Mangrove', + 'Project size': 'something', + 'Validation standard / accrediting organization': 'Verra', + 'Number of local individuals who own, work, and/or live in the project site (e.g., land tenure)': + 'land tenure', + 'City / province / region / state of project': 'Azpeiti', + 'Intended alternative use of land': 'commercial', + 'Land ownership before project': 'private', + 'SDGs benefitted / co-benefitted': 'sdg', + 'Project eligible for voluntary carbon credits?': 'no', + 'Are you willing to speak with us about your carbon credit pricing?': 'yes', + 'Are you able to provide additional detailed cost documentation for the project?': + 'yes', + 'Cost categories ': 'Detailed activities...', + 'Establishing community engagement / buy-in': 1, + 'Conservation project planning & administration ': 2, + 'Carbon project planning & administration': 3, + 'Land cost': 4, + 'Financing cost': 5, + 'Materials (e.g., seeds, fertilizer, seedlings)': 6, + 'Materials (e.g., machinery, equipment, etc.)': 7, + 'Project labor / activity': 8, + 'Engineering / construction intervention': 9, + 'Ongoing community engagement': 10, + 'Other project running cost': 11, + 'Project monitoring': 12, + '1) Other cost (please specify activities)': 13, + '2) Other cost (please specify activities)': 14, + '3) Other cost (please specify activities)': 15, + 'Project site cumulative sequestration / carbon stock': 'something', + 'Please describe in detail the project activity (e.g., planted mangrove seedlings, set up perimeter around conservation area)': + 'question 1', + 'When you kicked off the project, how did you spend to engage the community...': + 'question 2', + 'How did you acquire the rights to establish the project on the land?...': + 'question 3', + 'What was the hourly wage rate paid for labor? How many hours worked for each activity?': + 'question 4', + 'Please describe the ongoing community engagement for your project.': + 'question 5', + 'Did you undertake any engineering / construction interventions for your project?': + 'question 6', +}; + +// Función para transformar el JSON +export function userDataMapJsonToEntity( + inputJson: Record, + userId: string, +): Partial { + return { + programName: inputJson['Program name (if willing to share)'], + intendedLengthOfProject: inputJson['Intended length of project'], + country: inputJson['Country'], + currency: inputJson['Currency'], + projectStartYear: inputJson['Project start year'], + projectActivity: inputJson['Project activity'], + ecosystem: inputJson['Ecosystem'], + projectSize: inputJson['Project size'], + validationStandard: + inputJson['Validation standard / accrediting organization'], + numberOfLocalIndividuals: + inputJson[ + 'Number of local individuals who own, work, and/or live in the project site (e.g., land tenure)' + ], + cityOrRegion: inputJson['City / province / region / state of project'], + intendedAlternativeUseOfLand: inputJson['Intended alternative use of land'], + landOwnershipBeforeProject: inputJson['Land ownership before project'], + sdgsBenefitted: inputJson['SDGs benefitted / co-benefitted'], + projectEligibleForCarbonCredits: + inputJson['Project eligible for voluntary carbon credits?'] === 'yes', + willingToSpeakAboutPricing: + inputJson[ + 'Are you willing to speak with us about your carbon credit pricing?' + ] === 'yes', + ableToProvideDetailedCostDocumentation: + inputJson[ + 'Are you able to provide additional detailed cost documentation for the project?' + ] === 'yes', + costCategories: inputJson['Cost categories '], + establishingCommunityEngagement: + inputJson['Establishing community engagement / buy-in'], + conservationProjectPlanning: + inputJson['Conservation project planning & administration '], + carbonProjectPlanning: + inputJson['Carbon project planning & administration'], + landCost: inputJson['Land cost'], + financingCost: inputJson['Financing cost'], + materialsSeedsFertilizer: + inputJson['Materials (e.g., seeds, fertilizer, seedlings)'], + materialsMachineryEquipment: + inputJson['Materials (e.g., machinery, equipment, etc.)'], + projectLaborActivity: inputJson['Project labor / activity'], + engineeringIntervention: + inputJson['Engineering / construction intervention'], + ongoingCommunityEngagement: inputJson['Ongoing community engagement'], + otherProjectRunningCost: inputJson['Other project running cost'], + projectMonitoring: inputJson['Project monitoring'], + otherCost1: inputJson['1) Other cost (please specify activities)'], + otherCost2: inputJson['2) Other cost (please specify activities)'], + otherCost3: inputJson['3) Other cost (please specify activities)'], + projectCumulativeSequestration: + inputJson['Project site cumulative sequestration / carbon stock'], + detailedProjectActivity: + inputJson[ + 'Please describe in detail the project activity (e.g., planted mangrove seedlings, set up perimeter around conservation area)' + ], + communityEngagementSpending: + inputJson[ + 'When you kicked off the project, how did you spend to engage the community...' + ], + landRightsAndEasements: + inputJson[ + 'How did you acquire the rights to establish the project on the land?...' + ], + hourlyWageRate: + inputJson[ + 'What was the hourly wage rate paid for labor? How many hours worked for each activity?' + ], + ongoingCommunityCompensation: + inputJson[ + 'Please describe the ongoing community engagement for your project.' + ], + engineeringDetails: + inputJson[ + 'Did you undertake any engineering / construction interventions for your project?' + ], + user: { id: userId } as any, + }; +} diff --git a/api/src/modules/import/services/xlsx.parser.ts b/api/src/modules/import/services/xlsx.parser.ts index d69ba019..33510e66 100644 --- a/api/src/modules/import/services/xlsx.parser.ts +++ b/api/src/modules/import/services/xlsx.parser.ts @@ -21,4 +21,56 @@ export class XlsxParser implements ExcelParserInterface { return parsedData; } + + async parseUserExcels(data: Buffer[]) { + const carbonInputs: WorkBook = read(data[0]); + const costInputs: WorkBook = read(data[1]); + const CARBON_INPUTS_SHEETS = ['Restoration', 'Conservation']; + const COST_INPUTS_SHEETS = ['Input']; + const parsedCarbonInputs: any = {}; + let parsedCostInputs: WorkSheet; + // + // for (const sheetName of CARBON_INPUTS_SHEETS) { + // const sheet: WorkSheet = carbonInputs.Sheets[sheetName]; + // const parsedSheet = utils.sheet_to_json(sheet, { + // raw: true, + // }); + // parsedCarbonInputs[sheetName] = parsedSheet; + // } + + for (const sheetName of COST_INPUTS_SHEETS) { + parsedCostInputs = costInputs.Sheets[sheetName]; + //= utils.sheet_to_json(sheet, { header: 4 }); + //parsedCostInputs[sheetName] = parsedSheet; + } + const result: Record = {}; + const keysToIgnore = [ + 'Input data into blue shade cells', + 'General information', + 'Project information', + ]; + + Object.keys(parsedCostInputs).forEach((cellKey) => { + if (!cellKey.startsWith('B')) return; // Ignore cells that are not in column B + + const questionCell = parsedCostInputs[cellKey]; + const question = questionCell?.v; + + if (question && !keysToIgnore.includes(question)) { + // Answer is in the column C or D of the same row + const rowIndex = cellKey.match(/\d+/)?.[0]; // extract row number from cell key + const answerCellKey = `C${rowIndex}`; + const answerCell = + parsedCostInputs[answerCellKey] || parsedCostInputs[`D${rowIndex}`]; + + const answer = answerCell?.v || 'No value provided'; + result[question] = answer; + } + }); + + return { + //carbonInputs: parsedCarbonInputs, + costInputs: result, + }; + } } diff --git a/api/src/modules/users/users.module.ts b/api/src/modules/users/users.module.ts index 64a02288..481721b1 100644 --- a/api/src/modules/users/users.module.ts +++ b/api/src/modules/users/users.module.ts @@ -4,9 +4,13 @@ import { TypeOrmModule } from '@nestjs/typeorm'; import { User } from '@shared/entities/users/user.entity'; import { UsersController } from '@api/modules/users/users.controller'; import { AuthModule } from '@api/modules/auth/auth.module'; +import { UserUploadedData } from '@shared/entities/user-project-data.entity'; @Module({ - imports: [TypeOrmModule.forFeature([User]), forwardRef(() => AuthModule)], + imports: [ + TypeOrmModule.forFeature([User, UserUploadedData]), + forwardRef(() => AuthModule), + ], providers: [UsersService], exports: [UsersService], controllers: [UsersController], diff --git a/shared/contracts/users.contract.ts b/shared/contracts/users.contract.ts index 58585e60..1de44175 100644 --- a/shared/contracts/users.contract.ts +++ b/shared/contracts/users.contract.ts @@ -3,7 +3,7 @@ import { generateEntityQuerySchema } from "@shared/schemas/query-param.schema"; import { User } from "@shared/entities/users/user.entity"; import { UserDto } from "@shared/dtos/users/user.dto"; import { z } from "zod"; -import { JSONAPIError } from '@shared/dtos/json-api.error'; +import { JSONAPIError } from "@shared/dtos/json-api.error"; import { ApiResponse } from "@shared/dtos/global/api-response.dto"; import { UpdateUserPasswordSchema } from "@shared/schemas/users/update-password.schema"; @@ -13,8 +13,8 @@ import { UpdateUserSchema } from "@shared/schemas/users/update-user.schema"; const contract = initContract(); export const usersContract = contract.router({ findMe: { - method: 'GET', - path: '/users/me', + method: "GET", + path: "/users/me", responses: { 200: contract.type>(), 401: contract.type(), @@ -58,4 +58,13 @@ export const usersContract = contract.router({ }, body: null, }, + + uploadData: { + method: "POST", + path: "/users/upload-data", + responses: { + 201: contract.type(), + }, + body: contract.type<{ thumbnail: File }>(), + }, }); diff --git a/shared/entities/user-project-data.entity.ts b/shared/entities/user-project-data.entity.ts new file mode 100644 index 00000000..7516b904 --- /dev/null +++ b/shared/entities/user-project-data.entity.ts @@ -0,0 +1,139 @@ +import { + Entity, + PrimaryGeneratedColumn, + Column, + ManyToOne, + JoinColumn, + BaseEntity, +} from "typeorm"; +import { User } from "@shared/entities/users/user.entity"; + +@Entity("user_uploaded_data") +export class UserUploadedData extends BaseEntity { + @PrimaryGeneratedColumn("uuid") + id: string; + + @ManyToOne(() => User, (user) => user.uploadedData, { nullable: false }) + @JoinColumn({ name: "user_id" }) + user: User; + + @Column({ type: "varchar", length: 255, nullable: true }) + programName: string; + + @Column({ type: "int", nullable: true }) + intendedLengthOfProject: number; + + @Column({ type: "varchar", length: 255, nullable: true }) + country: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + currency: string; + + @Column({ type: "int", nullable: true }) + projectStartYear: number; + + @Column({ type: "varchar", length: 255, nullable: true }) + projectActivity: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + ecosystem: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + projectSize: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + validationStandard: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + numberOfLocalIndividuals: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + cityOrRegion: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + intendedAlternativeUseOfLand: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + landOwnershipBeforeProject: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + sdgsBenefitted: string; + + @Column({ type: "boolean", nullable: true }) + projectEligibleForCarbonCredits: boolean; + + @Column({ type: "boolean", nullable: true }) + willingToSpeakAboutPricing: boolean; + + @Column({ type: "boolean", nullable: true }) + ableToProvideDetailedCostDocumentation: boolean; + + @Column({ type: "text", nullable: true }) + costCategories: string; + + @Column({ type: "int", nullable: true }) + establishingCommunityEngagement: number; + + @Column({ type: "int", nullable: true }) + conservationProjectPlanning: number; + + @Column({ type: "int", nullable: true }) + carbonProjectPlanning: number; + + @Column({ type: "int", nullable: true }) + landCost: number; + + @Column({ type: "int", nullable: true }) + financingCost: number; + + @Column({ type: "int", nullable: true }) + materialsSeedsFertilizer: number; + + @Column({ type: "int", nullable: true }) + materialsMachineryEquipment: number; + + @Column({ type: "int", nullable: true }) + projectLaborActivity: number; + + @Column({ type: "int", nullable: true }) + engineeringIntervention: number; + + @Column({ type: "int", nullable: true }) + ongoingCommunityEngagement: number; + + @Column({ type: "int", nullable: true }) + otherProjectRunningCost: number; + + @Column({ type: "int", nullable: true }) + projectMonitoring: number; + + @Column({ type: "int", nullable: true }) + otherCost1: number; + + @Column({ type: "int", nullable: true }) + otherCost2: number; + + @Column({ type: "int", nullable: true }) + otherCost3: number; + + @Column({ type: "text", nullable: true }) + projectCumulativeSequestration: string; + + @Column({ type: "text", nullable: true }) + detailedProjectActivity: string; + + @Column({ type: "text", nullable: true }) + communityEngagementSpending: string; + + @Column({ type: "text", nullable: true }) + landRightsAndEasements: string; + + @Column({ type: "text", nullable: true }) + hourlyWageRate: string; + + @Column({ type: "text", nullable: true }) + ongoingCommunityCompensation: string; + + @Column({ type: "text", nullable: true }) + engineeringDetails: string; +} diff --git a/shared/entities/users/user.entity.ts b/shared/entities/users/user.entity.ts index ec4ffcac..a5f78edb 100644 --- a/shared/entities/users/user.entity.ts +++ b/shared/entities/users/user.entity.ts @@ -4,9 +4,11 @@ import { Entity, PrimaryGeneratedColumn, BaseEntity, + OneToMany, } from "typeorm"; import { Exclude } from "class-transformer"; import { ROLES } from "@shared/entities/users/roles.enum"; +import { UserUploadedData } from "@shared/entities/user-project-data.entity"; // TODO: For future reference: // https://github.com/typeorm/typeorm/issues/2897 @@ -42,4 +44,10 @@ export class User extends BaseEntity { @CreateDateColumn({ name: "created_at", type: "timestamp" }) createdAt: Date; + + // @OneToMany( + // () => UserUploadedData, + // (userUploadedData) => userUploadedData.user, + // ) + // uploadedData: UserUploadedData[]; } diff --git a/shared/lib/db-entities.ts b/shared/lib/db-entities.ts index a19a0af1..ba7f82dc 100644 --- a/shared/lib/db-entities.ts +++ b/shared/lib/db-entities.ts @@ -30,6 +30,7 @@ import { BaseSize } from "@shared/entities/base-size.entity"; import { BaseIncrease } from "@shared/entities/base-increase.entity"; import { ModelAssumptions } from "@shared/entities/model-assumptions.entity"; import { CustomProject } from "@shared/entities/custom-project.entity"; +import { UserUploadedData } from "@shared/entities/user-project-data.entity"; export const COMMON_DATABASE_ENTITIES = [ User, @@ -64,4 +65,5 @@ export const COMMON_DATABASE_ENTITIES = [ BaseIncrease, ModelAssumptions, CustomProject, + UserUploadedData, ]; From 34d48f5a99f3e77f49f897cc31235de21c8bdbbd Mon Sep 17 00:00:00 2001 From: alexeh Date: Wed, 20 Nov 2024 05:25:13 +0100 Subject: [PATCH 2/5] delete locale options and rename resources with ID property --- admin/index.ts | 25 ++----------------------- 1 file changed, 2 insertions(+), 23 deletions(-) diff --git a/admin/index.ts b/admin/index.ts index fe9ba923..33d802be 100644 --- a/admin/index.ts +++ b/admin/index.ts @@ -90,33 +90,12 @@ const start = async () => { resource: Country, name: "Country", options: { + id: "Countries", parent: databaseNavigation, icon: "Globe", }, }, - { - resource: UserUploadedData, - name: "UserUploadedData", - options: { - navigation: { - name: "Data Management", - icon: "Database", - }, - }, - }, ], - locale: { - language: "en", - translations: { - en: { - labels: { - User: "Users", - Country: "Countries", - Project: "Projects", - }, - }, - }, - }, }); const adminRouter = AdminJSExpress.buildAuthenticatedRouter(admin, { @@ -126,7 +105,7 @@ const start = async () => { const router = AdminJSExpress.buildRouter(admin); - app.use(admin.options.rootPath, adminRouter); + app.use(admin.options.rootPath, router); app.listen(PORT, () => { console.log( From 8ddb2c9f893c1aeb39de4c27d127fcb5c5b5a5d6 Mon Sep 17 00:00:00 2001 From: alexeh Date: Wed, 20 Nov 2024 06:22:25 +0100 Subject: [PATCH 3/5] refactor user upload data to user uploaded cost inputs, add navigation in adminjs --- admin/datasource.ts | 4 ++-- admin/index.ts | 24 ++++++++++++++++++- api/src/modules/import/import.service.ts | 4 ++-- .../import/services/user-data-parser.ts | 4 ++-- api/src/modules/users/users.module.ts | 4 ++-- shared/entities/user-project-data.entity.ts | 6 ++--- shared/entities/users/user.entity.ts | 9 +++---- shared/lib/db-entities.ts | 4 ++-- 8 files changed, 39 insertions(+), 20 deletions(-) diff --git a/admin/datasource.ts b/admin/datasource.ts index cbb22fe7..e731ada1 100644 --- a/admin/datasource.ts +++ b/admin/datasource.ts @@ -29,12 +29,12 @@ import { ImplementationLaborCost } from "@shared/entities/cost-inputs/implementa import { BaseSize } from "@shared/entities/base-size.entity.js"; import { BaseIncrease } from "@shared/entities/base-increase.entity.js"; import { ModelAssumptions } from "@shared/entities/model-assumptions.entity.js"; -import { UserUploadedData } from "@shared/entities/user-project-data.entity.js"; +import { UserUploadCostInputs } from "@shared/entities/user-project-data.entity.js"; // TODO: If we import the COMMON_DATABASE_ENTITIES from shared, we get an error where DataSouce is not set for a given entity export const ADMINJS_ENTITIES = [ User, - UserUploadedData, + UserUploadCostInputs, ApiEventsEntity, Country, ProjectSize, diff --git a/admin/index.ts b/admin/index.ts index 33d802be..19e53ce4 100644 --- a/admin/index.ts +++ b/admin/index.ts @@ -33,7 +33,7 @@ import { ImplementationLaborCostResource } from "./resources/implementation-labo import { BaseSizeResource } from "./resources/base-size/base-size.resource.js"; import { BaseIncreaseResource } from "./resources/base-increase/base-increase.resource.js"; import { ModelAssumptionResource } from "./resources/model-assumptions/model-assumptions.resource.js"; -import { UserUploadedData } from "@shared/entities/user-project-data.entity.js"; +import { UserUploadCostInputs } from "@shared/entities/user-project-data.entity.js"; AdminJS.registerAdapter({ Database: AdminJSTypeorm.Database, @@ -60,6 +60,16 @@ const start = async () => { componentLoader, resources: [ UserResource, + { + resource: UserUploadCostInputs, + name: "UserUploadCostInputs", + options: { + navigation: { + name: "User Data", + icon: "File", + }, + }, + }, ProjectSizeResource, FeasibilityAnalysisResource, ConservationAndPlanningAdminResource, @@ -96,6 +106,18 @@ const start = async () => { }, }, ], + locale: { + language: "en", + translations: { + en: { + labels: { + User: "Users", + Country: "Countries", + Project: "Projects", + }, + }, + }, + }, }); const adminRouter = AdminJSExpress.buildAuthenticatedRouter(admin, { diff --git a/api/src/modules/import/import.service.ts b/api/src/modules/import/import.service.ts index 0834dd47..42ccc4e3 100644 --- a/api/src/modules/import/import.service.ts +++ b/api/src/modules/import/import.service.ts @@ -13,7 +13,7 @@ import { userDataInputJson, userDataMapJsonToEntity, } from '@api/modules/import/services/user-data-parser'; -import { UserUploadedData } from '@shared/entities/user-project-data.entity'; +import { UserUploadCostInputs } from '@shared/entities/user-project-data.entity'; @Injectable() export class ImportService { @@ -56,7 +56,7 @@ export class ImportService { const { costInputs } = await this.excelParser.parseUserExcels(fileBuffers); const mapped = userDataMapJsonToEntity(userDataInputJson, userId); const savedData = await this.dataSource - .getRepository(UserUploadedData) + .getRepository(UserUploadCostInputs) .save(mapped); return savedData; } diff --git a/api/src/modules/import/services/user-data-parser.ts b/api/src/modules/import/services/user-data-parser.ts index a4f109f4..0cc68e10 100644 --- a/api/src/modules/import/services/user-data-parser.ts +++ b/api/src/modules/import/services/user-data-parser.ts @@ -1,6 +1,6 @@ // I feel dirty doing this... -import { UserUploadedData } from '@shared/entities/user-project-data.entity'; +import { UserUploadCostInputs } from '@shared/entities/user-project-data.entity'; export const userDataInputJson = { 'Program name (if willing to share)': 'test name', @@ -57,7 +57,7 @@ export const userDataInputJson = { export function userDataMapJsonToEntity( inputJson: Record, userId: string, -): Partial { +): Partial { return { programName: inputJson['Program name (if willing to share)'], intendedLengthOfProject: inputJson['Intended length of project'], diff --git a/api/src/modules/users/users.module.ts b/api/src/modules/users/users.module.ts index 481721b1..075802af 100644 --- a/api/src/modules/users/users.module.ts +++ b/api/src/modules/users/users.module.ts @@ -4,11 +4,11 @@ import { TypeOrmModule } from '@nestjs/typeorm'; import { User } from '@shared/entities/users/user.entity'; import { UsersController } from '@api/modules/users/users.controller'; import { AuthModule } from '@api/modules/auth/auth.module'; -import { UserUploadedData } from '@shared/entities/user-project-data.entity'; +import { UserUploadCostInputs } from '@shared/entities/user-project-data.entity'; @Module({ imports: [ - TypeOrmModule.forFeature([User, UserUploadedData]), + TypeOrmModule.forFeature([User, UserUploadCostInputs]), forwardRef(() => AuthModule), ], providers: [UsersService], diff --git a/shared/entities/user-project-data.entity.ts b/shared/entities/user-project-data.entity.ts index 7516b904..48e015c5 100644 --- a/shared/entities/user-project-data.entity.ts +++ b/shared/entities/user-project-data.entity.ts @@ -8,12 +8,12 @@ import { } from "typeorm"; import { User } from "@shared/entities/users/user.entity"; -@Entity("user_uploaded_data") -export class UserUploadedData extends BaseEntity { +@Entity("user_upload_cost_inputs") +export class UserUploadCostInputs extends BaseEntity { @PrimaryGeneratedColumn("uuid") id: string; - @ManyToOne(() => User, (user) => user.uploadedData, { nullable: false }) + @ManyToOne(() => User, (user) => user.uploadedData) @JoinColumn({ name: "user_id" }) user: User; diff --git a/shared/entities/users/user.entity.ts b/shared/entities/users/user.entity.ts index a5f78edb..72d7f7f2 100644 --- a/shared/entities/users/user.entity.ts +++ b/shared/entities/users/user.entity.ts @@ -8,7 +8,7 @@ import { } from "typeorm"; import { Exclude } from "class-transformer"; import { ROLES } from "@shared/entities/users/roles.enum"; -import { UserUploadedData } from "@shared/entities/user-project-data.entity"; +import { UserUploadCostInputs } from "@shared/entities/user-project-data.entity"; // TODO: For future reference: // https://github.com/typeorm/typeorm/issues/2897 @@ -45,9 +45,6 @@ export class User extends BaseEntity { @CreateDateColumn({ name: "created_at", type: "timestamp" }) createdAt: Date; - // @OneToMany( - // () => UserUploadedData, - // (userUploadedData) => userUploadedData.user, - // ) - // uploadedData: UserUploadedData[]; + @OneToMany("UserUploadCostInputs", "user") + uploadedData: UserUploadCostInputs[]; } diff --git a/shared/lib/db-entities.ts b/shared/lib/db-entities.ts index ba7f82dc..2f44e734 100644 --- a/shared/lib/db-entities.ts +++ b/shared/lib/db-entities.ts @@ -30,7 +30,7 @@ import { BaseSize } from "@shared/entities/base-size.entity"; import { BaseIncrease } from "@shared/entities/base-increase.entity"; import { ModelAssumptions } from "@shared/entities/model-assumptions.entity"; import { CustomProject } from "@shared/entities/custom-project.entity"; -import { UserUploadedData } from "@shared/entities/user-project-data.entity"; +import { UserUploadCostInputs } from "@shared/entities/user-project-data.entity"; export const COMMON_DATABASE_ENTITIES = [ User, @@ -65,5 +65,5 @@ export const COMMON_DATABASE_ENTITIES = [ BaseIncrease, ModelAssumptions, CustomProject, - UserUploadedData, + UserUploadCostInputs, ]; From 9128904d4cb054073ff44d66763d8482eea7517c Mon Sep 17 00:00:00 2001 From: alexeh Date: Wed, 20 Nov 2024 08:05:07 +0100 Subject: [PATCH 4/5] upload user cost forms and list in backoffice --- admin/datasource.ts | 6 +- admin/index.ts | 25 +++- api/src/modules/import/import.controller.ts | 3 - api/src/modules/import/import.service.ts | 44 +++++-- .../import/services/user-data-parser.ts | 118 ++++++++++-------- .../modules/import/services/xlsx.parser.ts | 93 ++++++++++---- api/src/modules/users/users.module.ts | 11 +- .../user-upload-conservation-inputs.entity.ts | 73 +++++++++++ .../user-upload-cost-inputs.entity.ts} | 2 +- .../user-upload-restoration-inputs.entity.ts | 73 +++++++++++ shared/entities/users/user.entity.ts | 12 +- shared/lib/db-entities.ts | 6 +- 12 files changed, 366 insertions(+), 100 deletions(-) create mode 100644 shared/entities/users/user-upload-conservation-inputs.entity.ts rename shared/entities/{user-project-data.entity.ts => users/user-upload-cost-inputs.entity.ts} (98%) create mode 100644 shared/entities/users/user-upload-restoration-inputs.entity.ts diff --git a/admin/datasource.ts b/admin/datasource.ts index e731ada1..161ef3d4 100644 --- a/admin/datasource.ts +++ b/admin/datasource.ts @@ -29,12 +29,16 @@ import { ImplementationLaborCost } from "@shared/entities/cost-inputs/implementa import { BaseSize } from "@shared/entities/base-size.entity.js"; import { BaseIncrease } from "@shared/entities/base-increase.entity.js"; import { ModelAssumptions } from "@shared/entities/model-assumptions.entity.js"; -import { UserUploadCostInputs } from "@shared/entities/user-project-data.entity.js"; +import { UserUploadCostInputs } from "@shared/entities/users/user-upload-cost-inputs.entity.js"; +import { UserUploadRestorationInputs } from "@shared/entities/users/user-upload-restoration-inputs.entity.js"; +import { UserUploadConservationInputs } from "@shared/entities/users/user-upload-conservation-inputs.entity.js"; // TODO: If we import the COMMON_DATABASE_ENTITIES from shared, we get an error where DataSouce is not set for a given entity export const ADMINJS_ENTITIES = [ User, UserUploadCostInputs, + UserUploadRestorationInputs, + UserUploadConservationInputs, ApiEventsEntity, Country, ProjectSize, diff --git a/admin/index.ts b/admin/index.ts index 19e53ce4..585e839d 100644 --- a/admin/index.ts +++ b/admin/index.ts @@ -33,7 +33,9 @@ import { ImplementationLaborCostResource } from "./resources/implementation-labo import { BaseSizeResource } from "./resources/base-size/base-size.resource.js"; import { BaseIncreaseResource } from "./resources/base-increase/base-increase.resource.js"; import { ModelAssumptionResource } from "./resources/model-assumptions/model-assumptions.resource.js"; -import { UserUploadCostInputs } from "@shared/entities/user-project-data.entity.js"; +import { UserUploadCostInputs } from "@shared/entities/users/user-upload-cost-inputs.entity.js"; +import { UserUploadConservationInputs } from "@shared/entities/users/user-upload-conservation-inputs.entity.js"; +import { UserUploadRestorationInputs } from "@shared/entities/users/user-upload-restoration-inputs.entity.js"; AdminJS.registerAdapter({ Database: AdminJSTypeorm.Database, @@ -70,6 +72,26 @@ const start = async () => { }, }, }, + { + resource: UserUploadConservationInputs, + name: "UserUploadConservationInputs", + options: { + navigation: { + name: "User Data", + icon: "File", + }, + }, + }, + { + resource: UserUploadRestorationInputs, + name: "UserUploadRestorationInputs", + options: { + navigation: { + name: "User Data", + icon: "File", + }, + }, + }, ProjectSizeResource, FeasibilityAnalysisResource, ConservationAndPlanningAdminResource, @@ -100,7 +122,6 @@ const start = async () => { resource: Country, name: "Country", options: { - id: "Countries", parent: databaseNavigation, icon: "Globe", }, diff --git a/api/src/modules/import/import.controller.ts b/api/src/modules/import/import.controller.ts index cd0b4d28..9a4292fd 100644 --- a/api/src/modules/import/import.controller.ts +++ b/api/src/modules/import/import.controller.ts @@ -20,7 +20,6 @@ import { Multer } from 'multer'; import { GetUser } from '@api/modules/auth/decorators/get-user.decorator'; import { User } from '@shared/entities/users/user.entity'; import { usersContract } from '@shared/contracts/users.contract'; -import { Public } from '@api/modules/auth/decorators/is-public.decorator'; @Controller() @UseGuards(JwtAuthGuard, RolesGuard) @@ -45,7 +44,6 @@ export class ImportController { }); } - //@Public() @UseInterceptors(FilesInterceptor('files', 2)) @RequiredRoles(ROLES.PARTNER, ROLES.ADMIN) @TsRestHandler(usersContract.uploadData) @@ -54,7 +52,6 @@ export class ImportController { @UploadedFiles() files: Array, ): Promise { return tsRestHandler(usersContract.uploadData, async () => { - console.log(files); const [file1, file2] = files; const [file1Buffer, file2Buffer] = [file1.buffer, file2.buffer]; const data = await this.service.importDataProvidedByPartner( diff --git a/api/src/modules/import/import.service.ts b/api/src/modules/import/import.service.ts index 42ccc4e3..b4f5c57f 100644 --- a/api/src/modules/import/import.service.ts +++ b/api/src/modules/import/import.service.ts @@ -10,10 +10,13 @@ import { API_EVENT_TYPES } from '@api/modules/api-events/events.enum'; import { ImportEvent } from '@api/modules/import/events/import.event'; import { DataSource } from 'typeorm'; import { - userDataInputJson, - userDataMapJsonToEntity, + userDataConservationInputMapJsonToEntity, + userDataCostInputsMapJsonToEntity, + userDataRestorationInputMapJsonToEntity, } from '@api/modules/import/services/user-data-parser'; -import { UserUploadCostInputs } from '@shared/entities/user-project-data.entity'; +import { UserUploadCostInputs } from '@shared/entities/users/user-upload-cost-inputs.entity'; +import { UserUploadRestorationInputs } from '@shared/entities/users/user-upload-restoration-inputs.entity'; +import { UserUploadConservationInputs } from '@shared/entities/users/user-upload-conservation-inputs.entity'; @Injectable() export class ImportService { @@ -53,11 +56,34 @@ export class ImportService { } async importDataProvidedByPartner(fileBuffers: Buffer[], userId: string) { - const { costInputs } = await this.excelParser.parseUserExcels(fileBuffers); - const mapped = userDataMapJsonToEntity(userDataInputJson, userId); - const savedData = await this.dataSource - .getRepository(UserUploadCostInputs) - .save(mapped); - return savedData; + // TODO: Debt, add event handling + const { costInputs, carbonInputs } = + await this.excelParser.parseUserExcels(fileBuffers); + const mappedCostInputs = userDataCostInputsMapJsonToEntity( + costInputs, + userId, + ); + const mappedRestorationInputs = userDataRestorationInputMapJsonToEntity( + carbonInputs.restoration, + userId, + ); + const mappedConservationInputs = userDataConservationInputMapJsonToEntity( + carbonInputs.conservation, + userId, + ); + await this.dataSource.transaction(async (manager) => { + const userCostInputsRepo = manager.getRepository(UserUploadCostInputs); + const userRestorationInputsRepo = manager.getRepository( + UserUploadRestorationInputs, + ); + const userConservationInputsRepo = manager.getRepository( + UserUploadConservationInputs, + ); + await userCostInputsRepo.save(mappedCostInputs); + await userRestorationInputsRepo.save(mappedRestorationInputs); + await userConservationInputsRepo.save(mappedConservationInputs); + }); + // + return carbonInputs; } } diff --git a/api/src/modules/import/services/user-data-parser.ts b/api/src/modules/import/services/user-data-parser.ts index 0cc68e10..edc711fa 100644 --- a/api/src/modules/import/services/user-data-parser.ts +++ b/api/src/modules/import/services/user-data-parser.ts @@ -1,60 +1,10 @@ // I feel dirty doing this... -import { UserUploadCostInputs } from '@shared/entities/user-project-data.entity'; +import { UserUploadCostInputs } from '@shared/entities/users/user-upload-cost-inputs.entity'; +import { UserUploadRestorationInputs } from '@shared/entities/users/user-upload-restoration-inputs.entity'; +import { UserUploadConservationInputs } from '@shared/entities/users/user-upload-conservation-inputs.entity'; -export const userDataInputJson = { - 'Program name (if willing to share)': 'test name', - 'Intended length of project': 2000, - Country: 'Basque Country', - Currency: 'Euro', - 'Project start year': 1, - 'Project activity': 'Conservation', - Ecosystem: 'Mangrove', - 'Project size': 'something', - 'Validation standard / accrediting organization': 'Verra', - 'Number of local individuals who own, work, and/or live in the project site (e.g., land tenure)': - 'land tenure', - 'City / province / region / state of project': 'Azpeiti', - 'Intended alternative use of land': 'commercial', - 'Land ownership before project': 'private', - 'SDGs benefitted / co-benefitted': 'sdg', - 'Project eligible for voluntary carbon credits?': 'no', - 'Are you willing to speak with us about your carbon credit pricing?': 'yes', - 'Are you able to provide additional detailed cost documentation for the project?': - 'yes', - 'Cost categories ': 'Detailed activities...', - 'Establishing community engagement / buy-in': 1, - 'Conservation project planning & administration ': 2, - 'Carbon project planning & administration': 3, - 'Land cost': 4, - 'Financing cost': 5, - 'Materials (e.g., seeds, fertilizer, seedlings)': 6, - 'Materials (e.g., machinery, equipment, etc.)': 7, - 'Project labor / activity': 8, - 'Engineering / construction intervention': 9, - 'Ongoing community engagement': 10, - 'Other project running cost': 11, - 'Project monitoring': 12, - '1) Other cost (please specify activities)': 13, - '2) Other cost (please specify activities)': 14, - '3) Other cost (please specify activities)': 15, - 'Project site cumulative sequestration / carbon stock': 'something', - 'Please describe in detail the project activity (e.g., planted mangrove seedlings, set up perimeter around conservation area)': - 'question 1', - 'When you kicked off the project, how did you spend to engage the community...': - 'question 2', - 'How did you acquire the rights to establish the project on the land?...': - 'question 3', - 'What was the hourly wage rate paid for labor? How many hours worked for each activity?': - 'question 4', - 'Please describe the ongoing community engagement for your project.': - 'question 5', - 'Did you undertake any engineering / construction interventions for your project?': - 'question 6', -}; - -// Función para transformar el JSON -export function userDataMapJsonToEntity( +export function userDataCostInputsMapJsonToEntity( inputJson: Record, userId: string, ): Partial { @@ -138,3 +88,63 @@ export function userDataMapJsonToEntity( user: { id: userId } as any, }; } + +export function userDataRestorationInputMapJsonToEntity( + data: Record, + userId: string, +): Partial { + return { + projectName: data['Project name'], + country: data['Country'], + cityOrRegion: data['City / province / region / state of project'], + projectStartYear: data['Project start year'], + mostRecentYearOfData: data['Most recent year of data collection'], + ecosystem: data['Ecosystem'], + projectActivity: data['Project activity']?.toString(), // Convertir a string si es numérico + projectSizeAreaStudied: data['Project size / area studied'], + categories: data['Categories'], + projectArea: data['Project area '], + abovegroundBiomassStock: data['Aboveground biomass stock'], + belowgroundBiomassStock: data['Belowground biomass stock '], + soilOrganicCarbonStock: data['Soil organic carbon stock '], + methaneEmissions: data['Methane emissions '], + nitrousOxideEmissions: data['Nitrous oxide emissions '], + abovegroundBiomassEmissionsFactor: + data['Aboveground biomass emissions factor'], + belowgroundBiomassEmissionsFactor: + data['Belowground biomass emissions factor'], + soilOrganicCarbonEmissionsFactor: + data['Soil organic carbon emissions factor '], + user: { id: userId } as any, + }; +} + +export function userDataConservationInputMapJsonToEntity( + data: Record, + userId: string, +): Partial { + return { + projectName: data['Project name'], + country: data['Country'], + cityOrRegion: data['City / province / region / state of project'], + projectStartYear: data['Project start year'], + mostRecentYearOfData: data['Most recent year of data collection'], + ecosystem: data['Ecosystem'], + projectActivity: data['Project activity']?.toString(), // Convertir a string si es numérico + projectSizeAreaStudied: data['Project size / area studied'], + categories: data['Categories'], + projectArea: data['Project area '], + abovegroundBiomassStock: data['Aboveground biomass stock'], + belowgroundBiomassStock: data['Belowground biomass stock '], + soilOrganicCarbonStock: data['Soil organic carbon stock '], + methaneEmissions: data['Methane emissions '], + nitrousOxideEmissions: data['Nitrous oxide emissions '], + abovegroundBiomassEmissionsFactor: + data['Aboveground biomass emissions factor'], + belowgroundBiomassEmissionsFactor: + data['Belowground biomass emissions factor'], + soilOrganicCarbonEmissionsFactor: + data['Soil organic carbon emissions factor '], + user: { id: userId } as any, + }; +} diff --git a/api/src/modules/import/services/xlsx.parser.ts b/api/src/modules/import/services/xlsx.parser.ts index 33510e66..caf2d698 100644 --- a/api/src/modules/import/services/xlsx.parser.ts +++ b/api/src/modules/import/services/xlsx.parser.ts @@ -25,35 +25,24 @@ export class XlsxParser implements ExcelParserInterface { async parseUserExcels(data: Buffer[]) { const carbonInputs: WorkBook = read(data[0]); const costInputs: WorkBook = read(data[1]); - const CARBON_INPUTS_SHEETS = ['Restoration', 'Conservation']; - const COST_INPUTS_SHEETS = ['Input']; - const parsedCarbonInputs: any = {}; - let parsedCostInputs: WorkSheet; - // - // for (const sheetName of CARBON_INPUTS_SHEETS) { - // const sheet: WorkSheet = carbonInputs.Sheets[sheetName]; - // const parsedSheet = utils.sheet_to_json(sheet, { - // raw: true, - // }); - // parsedCarbonInputs[sheetName] = parsedSheet; - // } - - for (const sheetName of COST_INPUTS_SHEETS) { - parsedCostInputs = costInputs.Sheets[sheetName]; - //= utils.sheet_to_json(sheet, { header: 4 }); - //parsedCostInputs[sheetName] = parsedSheet; - } - const result: Record = {}; + + const restorationSheet: WorkSheet = carbonInputs.Sheets['Restoration']; + const conservationSheet: WorkSheet = carbonInputs.Sheets['Conservation']; + const restoration = parseRestorationSheet(restorationSheet); + const costInputSheet = costInputs.Sheets['Input']; + const conservation = parseConservationSheet(conservationSheet); + + const costInput: Record = {}; const keysToIgnore = [ 'Input data into blue shade cells', 'General information', 'Project information', ]; - Object.keys(parsedCostInputs).forEach((cellKey) => { + Object.keys(costInputSheet).forEach((cellKey) => { if (!cellKey.startsWith('B')) return; // Ignore cells that are not in column B - const questionCell = parsedCostInputs[cellKey]; + const questionCell = costInputSheet[cellKey]; const question = questionCell?.v; if (question && !keysToIgnore.includes(question)) { @@ -61,16 +50,70 @@ export class XlsxParser implements ExcelParserInterface { const rowIndex = cellKey.match(/\d+/)?.[0]; // extract row number from cell key const answerCellKey = `C${rowIndex}`; const answerCell = - parsedCostInputs[answerCellKey] || parsedCostInputs[`D${rowIndex}`]; + costInputSheet[answerCellKey] || costInputSheet[`D${rowIndex}`]; const answer = answerCell?.v || 'No value provided'; - result[question] = answer; + costInput[question] = answer; } }); return { - //carbonInputs: parsedCarbonInputs, - costInputs: result, + carbonInputs: { restoration, conservation }, + costInputs: costInput, }; } } + +function parseRestorationSheet(sheet: WorkSheet): Record { + const result: Record = {}; + + const keysToIgnore = [ + 'Sub-national / project sequestration information', + 'General information', + ]; + + Object.keys(sheet).forEach((cellKey) => { + if (!cellKey.startsWith('B')) return; + + const questionCell = sheet[cellKey]; + const question = questionCell?.v; + + if (question && !keysToIgnore.includes(question)) { + const rowIndex = cellKey.match(/\d+/)?.[0]; + const answerCellKey = `C${rowIndex}`; + const answerCell = sheet[answerCellKey]; + + const answer = answerCell?.v || 'No value provided'; + result[question] = answer; + } + }); + + return result; +} + +function parseConservationSheet(sheet: WorkSheet): Record { + const result: Record = {}; + + const keysToIgnore = [ + 'Sub-national / project sequestration information', + 'General information', + ]; + + Object.keys(sheet).forEach((cellKey) => { + if (!cellKey.startsWith('B')) return; + + const questionCell = sheet[cellKey]; + const question = questionCell?.v; + + if (question && !keysToIgnore.includes(question)) { + const rowIndex = cellKey.match(/\d+/)?.[0]; + const answerCellKey = `C${rowIndex}`; + const answerCell = sheet[answerCellKey]; + + const answer = answerCell?.v || 'No value provided'; + result[question] = answer; + } + }); + + return result; +} diff --git a/api/src/modules/users/users.module.ts b/api/src/modules/users/users.module.ts index 075802af..9810333c 100644 --- a/api/src/modules/users/users.module.ts +++ b/api/src/modules/users/users.module.ts @@ -4,11 +4,18 @@ import { TypeOrmModule } from '@nestjs/typeorm'; import { User } from '@shared/entities/users/user.entity'; import { UsersController } from '@api/modules/users/users.controller'; import { AuthModule } from '@api/modules/auth/auth.module'; -import { UserUploadCostInputs } from '@shared/entities/user-project-data.entity'; +import { UserUploadCostInputs } from '@shared/entities/users/user-upload-cost-inputs.entity'; +import { UserUploadRestorationInputs } from '@shared/entities/users/user-upload-restoration-inputs.entity'; +import { UserUploadConservationInputs } from '@shared/entities/users/user-upload-conservation-inputs.entity'; @Module({ imports: [ - TypeOrmModule.forFeature([User, UserUploadCostInputs]), + TypeOrmModule.forFeature([ + User, + UserUploadCostInputs, + UserUploadRestorationInputs, + UserUploadConservationInputs, + ]), forwardRef(() => AuthModule), ], providers: [UsersService], diff --git a/shared/entities/users/user-upload-conservation-inputs.entity.ts b/shared/entities/users/user-upload-conservation-inputs.entity.ts new file mode 100644 index 00000000..429065bd --- /dev/null +++ b/shared/entities/users/user-upload-conservation-inputs.entity.ts @@ -0,0 +1,73 @@ +import { + Entity, + PrimaryGeneratedColumn, + Column, + BaseEntity, + ManyToOne, + JoinColumn, +} from "typeorm"; +import { User } from "@shared/entities/users/user.entity"; + +@Entity("user_upload_conservation_inputs") +export class UserUploadConservationInputs extends BaseEntity { + @PrimaryGeneratedColumn() + id: number; + + @ManyToOne(() => User, (user) => user.userUploadConservationInputs) + @JoinColumn({ name: "user_id" }) + user: User; + + @Column({ type: "varchar", length: 255, nullable: true }) + projectName: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + country: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + cityOrRegion: string; + + @Column({ type: "int", nullable: true }) + projectStartYear: number; + + @Column({ type: "int", nullable: true }) + mostRecentYearOfData: number; + + @Column({ type: "varchar", length: 255, nullable: true }) + ecosystem: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + projectActivity: string; + + @Column({ type: "int", nullable: true }) + projectSizeAreaStudied: number; + + @Column({ type: "varchar", length: 255, nullable: true }) + categories: string; + + @Column({ type: "int", nullable: true }) + projectArea: number; + + @Column({ type: "int", nullable: true }) + abovegroundBiomassStock: number; + + @Column({ type: "int", nullable: true }) + belowgroundBiomassStock: number; + + @Column({ type: "int", nullable: true }) + soilOrganicCarbonStock: number; + + @Column({ type: "int", nullable: true }) + methaneEmissions: number; + + @Column({ type: "int", nullable: true }) + nitrousOxideEmissions: number; + + @Column({ type: "int", nullable: true }) + abovegroundBiomassEmissionsFactor: number; + + @Column({ type: "int", nullable: true }) + belowgroundBiomassEmissionsFactor: number; + + @Column({ type: "int", nullable: true }) + soilOrganicCarbonEmissionsFactor: number; +} diff --git a/shared/entities/user-project-data.entity.ts b/shared/entities/users/user-upload-cost-inputs.entity.ts similarity index 98% rename from shared/entities/user-project-data.entity.ts rename to shared/entities/users/user-upload-cost-inputs.entity.ts index 48e015c5..16e97e66 100644 --- a/shared/entities/user-project-data.entity.ts +++ b/shared/entities/users/user-upload-cost-inputs.entity.ts @@ -13,7 +13,7 @@ export class UserUploadCostInputs extends BaseEntity { @PrimaryGeneratedColumn("uuid") id: string; - @ManyToOne(() => User, (user) => user.uploadedData) + @ManyToOne(() => User, (user) => user.uploadedCostInputs) @JoinColumn({ name: "user_id" }) user: User; diff --git a/shared/entities/users/user-upload-restoration-inputs.entity.ts b/shared/entities/users/user-upload-restoration-inputs.entity.ts new file mode 100644 index 00000000..be43501a --- /dev/null +++ b/shared/entities/users/user-upload-restoration-inputs.entity.ts @@ -0,0 +1,73 @@ +import { + Entity, + PrimaryGeneratedColumn, + Column, + BaseEntity, + ManyToOne, + JoinColumn, +} from "typeorm"; +import { User } from "@shared/entities/users/user.entity"; + +@Entity("user_upload_restoration_inputs") +export class UserUploadRestorationInputs extends BaseEntity { + @PrimaryGeneratedColumn() + id: number; + + @ManyToOne(() => User, (user) => user.userUploadRestorationInputs) + @JoinColumn({ name: "user_id" }) + user: User; + + @Column({ type: "varchar", length: 255, nullable: true }) + projectName: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + country: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + cityOrRegion: string; + + @Column({ type: "int", nullable: true }) + projectStartYear: number; + + @Column({ type: "int", nullable: true }) + mostRecentYearOfData: number; + + @Column({ type: "varchar", length: 255, nullable: true }) + ecosystem: string; + + @Column({ type: "varchar", length: 255, nullable: true }) + projectActivity: string; + + @Column({ type: "int", nullable: true }) + projectSizeAreaStudied: number; + + @Column({ type: "varchar", length: 255, nullable: true }) + categories: string; + + @Column({ type: "int", nullable: true }) + projectArea: number; + + @Column({ type: "int", nullable: true }) + abovegroundBiomassStock: number; + + @Column({ type: "int", nullable: true }) + belowgroundBiomassStock: number; + + @Column({ type: "int", nullable: true }) + soilOrganicCarbonStock: number; + + @Column({ type: "int", nullable: true }) + methaneEmissions: number; + + @Column({ type: "int", nullable: true }) + nitrousOxideEmissions: number; + + @Column({ type: "int", nullable: true }) + abovegroundBiomassEmissionsFactor: number; + + @Column({ type: "int", nullable: true }) + belowgroundBiomassEmissionsFactor: number; + + @Column({ type: "int", nullable: true }) + soilOrganicCarbonEmissionsFactor: number; +} diff --git a/shared/entities/users/user.entity.ts b/shared/entities/users/user.entity.ts index 72d7f7f2..16727524 100644 --- a/shared/entities/users/user.entity.ts +++ b/shared/entities/users/user.entity.ts @@ -8,7 +8,9 @@ import { } from "typeorm"; import { Exclude } from "class-transformer"; import { ROLES } from "@shared/entities/users/roles.enum"; -import { UserUploadCostInputs } from "@shared/entities/user-project-data.entity"; +import { UserUploadCostInputs } from "@shared/entities/users/user-upload-cost-inputs.entity"; +import { UserUploadRestorationInputs } from "@shared/entities/users/user-upload-restoration-inputs.entity"; +import { UserUploadConservationInputs } from "@shared/entities/users/user-upload-conservation-inputs.entity"; // TODO: For future reference: // https://github.com/typeorm/typeorm/issues/2897 @@ -46,5 +48,11 @@ export class User extends BaseEntity { createdAt: Date; @OneToMany("UserUploadCostInputs", "user") - uploadedData: UserUploadCostInputs[]; + uploadedCostInputs: UserUploadCostInputs[]; + + @OneToMany("UserUploadRestorationInputs", "user") + userUploadRestorationInputs: UserUploadRestorationInputs[]; + + @OneToMany("UserUploadConservationInputs", "user") + userUploadConservationInputs: UserUploadConservationInputs[]; } diff --git a/shared/lib/db-entities.ts b/shared/lib/db-entities.ts index 2f44e734..dbc802fc 100644 --- a/shared/lib/db-entities.ts +++ b/shared/lib/db-entities.ts @@ -30,7 +30,9 @@ import { BaseSize } from "@shared/entities/base-size.entity"; import { BaseIncrease } from "@shared/entities/base-increase.entity"; import { ModelAssumptions } from "@shared/entities/model-assumptions.entity"; import { CustomProject } from "@shared/entities/custom-project.entity"; -import { UserUploadCostInputs } from "@shared/entities/user-project-data.entity"; +import { UserUploadCostInputs } from "@shared/entities/users/user-upload-cost-inputs.entity"; +import { UserUploadRestorationInputs } from "@shared/entities/users/user-upload-restoration-inputs.entity"; +import { UserUploadConservationInputs } from "@shared/entities/users/user-upload-conservation-inputs.entity"; export const COMMON_DATABASE_ENTITIES = [ User, @@ -66,4 +68,6 @@ export const COMMON_DATABASE_ENTITIES = [ ModelAssumptions, CustomProject, UserUploadCostInputs, + UserUploadRestorationInputs, + UserUploadConservationInputs, ]; From 00fb87951687659369697d5baedb57edeb9bb91d Mon Sep 17 00:00:00 2001 From: alexeh Date: Wed, 20 Nov 2024 09:25:40 +0100 Subject: [PATCH 5/5] add user templates --- admin/index.ts | 2 +- .../src/public/forms/Carbon input template.xlsx | Bin 0 -> 33900 bytes .../src/public/forms/Cost input template.xlsx | Bin 0 -> 37390 bytes 3 files changed, 1 insertion(+), 1 deletion(-) create mode 100644 client/src/public/forms/Carbon input template.xlsx create mode 100644 client/src/public/forms/Cost input template.xlsx diff --git a/admin/index.ts b/admin/index.ts index 585e839d..25a33b38 100644 --- a/admin/index.ts +++ b/admin/index.ts @@ -148,7 +148,7 @@ const start = async () => { const router = AdminJSExpress.buildRouter(admin); - app.use(admin.options.rootPath, router); + app.use(admin.options.rootPath, adminRouter); app.listen(PORT, () => { console.log( diff --git a/client/src/public/forms/Carbon input template.xlsx b/client/src/public/forms/Carbon input template.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..1f8fb4321992b81c6bdb49ee78f51ad232189fd1 GIT binary patch literal 33900 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 Oeq^5lwptE9KK&ndh~~2Z literal 0 HcmV?d00001 diff --git a/client/src/public/forms/Cost input template.xlsx b/client/src/public/forms/Cost input template.xlsx new file mode 100644 index 0000000000000000000000000000000000000000..b47afea2ebc20d277a1a7ccd80680b7751cdca91 GIT binary patch literal 37390 zcmeFXbC)hpn};J)Y@PL#J?u@K zbm-h|tO*K0fGF|+fd01sf5-pe2n?!D*lsW&bfKT}L3E)Z*|btb{&9q}aFPZl0rfN) zOc7D{_V1Lv?4_6B6h$6o1c4kA34Be@$X;@6@7g?L^8$$*&rF<(!kqY{tV-2Lx0d^H z@d-W=ADmFo3?|tEZk$#*lbibD1)d(V1tztO&<;IaZ^;i$Qw5A{YI)I|cYC(78DUAh z{0F^p%}rNBNS~rKyiau^S3b%{YOPiW<2sp#VxXbYR;P$6!@fo(C66GP59=+Gq}8RG zD!yg?96%@Ufq;Y!plGAt0~O;77a`qJ7g%iPA`!> zS4YFdfpl+0xa)cZ@XAfRU@eJpYyU#bY#)7()mze27PuAr&<26fm2?Z5CdC8J1@@-} zP4O|6{GWX9Z{P56cE%DvTBRg{7 zX%|6PgWRF3iPVQY6UF?cV@)-5hgxquUhzas9Kg7oG>3?V+O;*cskN(sr84yvJ}8t! zLxOCR-wNer*i8t`_owrcB7H!m24yMW-9cDOL^C%kuNj1DLDKb%Au!lZ#3CM#XNOV^ zDI#FNFp^4tGTL zZoid^z;=wE{vzeSV#w6oHdo^>en|i7>R*rnxLeb^**RJn+SytCOGJxQbnLPu5qx?T ze}LUSq}3VZu|d`P;5t$>DI_-!Nfny|hN^%%oiz*#Dt^AE@Ap<+T#mAcybSm_oJ@}= z?Z4Y;p$|gFGa>PaK46H0?uCn@?F1iPaI=(*7b9-d(0EJcX%~ly)VFa~VxPZ+o$$bucV`w*iK?r!^1TDmCOU;Lkb7E^sVN5D8E1Gg1h0UpjdAylZoBlfqD@^@+~Q zc@E``NIdX-#U~<-aNPm=@rwO=sI3R6V8RG-FDwF_sXRg#AI9|vSo8;&&v#Tq_&96D zSGbh(A@5*d)YVSE?su75Gd|1AK^H-EJs1m#@e4$sher~?N`XIHPA7CEY!%Z|E5dcN z#$loHY5W0(M^7JkO*7d1DCb>@xyqTd`67eiWnBj!R;q_h;LPxZdwuB8N|(`=UvAm( z(=Q(vV##=?`+c_Zoo}54Y%${cNVMorhF$=yIFX|alU5!EthkI;G%)Sbymn!Uwnq1kLTnh3wCF`9gv+^fCq(Xbl#%L7U?CcV4yLc03eMp7Br zq@21T3*0k0Y-!7InI6W0qpW_{wOj{t`@1&C?e;AsJlmC{hwpB8|J~AOsrarHP94EW zd=$gN>a&052lSs%xU*?$LHZY;et+*`0YCu$MdAMvv;U06|1E5QfB)6~ZTmm_=t`Ec z8eo74y$SjX+x^%lx$3Z8JTrmu+~QhY;ur^yxzaD+xW6$2vO=TSCOtHa57y&7ZEoK1 zyI^|<35#J3EOzK8Vi-q@DF^Z5>)Eu?2quvs)Ux8Q!;8Oq*vQ(VZc6+lt;&KUt~Hde z%PjNS$5DaRG_>Ys=uy-xC4HM!=}I~$q2AR}{c) z5bbMJF@p`QY1%!HHhv1T?Jjdhjn{YJG4&A2Q>zy<{p6aml*PJN3QRDDc(X4d3=JzL zPYf|{;I!ze;x>TgKTxKlJQN~@NTP#5eza(8k`u`M;Iybqg^Of&^p)8?T_VBtWVz5V zD6{)g?f5ju*_ZzqDbCjTnZF%dpY;2?HvTKskKd&s5rYE&1mFMwVEk?IA5`yTZerr> zME~y-H(`M1MNSEQ89usY?9K4jz~h5{6V(BGeMUz_b47 z)#q2|n(A3|IiR(F@GweIOG~zBf&r_zrQnia5Od0WqNev!wteS|Jd!w@k0z_+ zLh+w2HwjUVx&`;rSC}}1&zC6iMCR(~3hVI0vIXyqZ<0*m<{Q>rVQQpZEDLM8on$g@>d)PWTZDjcUK~}RBJ*HfbT<)5wU#u?>zk;Qe|i2%WNb*4~vUH>3)(ZF$>vPCB3C*kJ4XdLehf{wCPiZvIh3SwCU=6bp|;66CW@J|8tSBFonf;;`+K z$5~J3=FRImjJ$lmWzl`waE=$!PIiLQG?Imb9%cp>GQK|w=FM;IGej(TXolx@YQ`jAd0G9hN z5*qYd&`O_3=?{}(U0YZkKEx2x$+$W$%=?Vh7@@Sj^mGH*I`lq#|?H$@{Qv7@>kEcd}_^TFr!n`aI+1x8R3RjcYG= z!(lTHm><1kQ59ztSI^0wap015=OV`tSfv?iu2laW;wmcrN#Y~swHkQc79MfMBHit+RUkozq|41tqC@6{ zl=|rvyv{V8sDT!3N}E7v^O-r)MMeEj%+_QEA?&1j;A?_3~5d8~K!S!YbK z9=#WV(_`a9v~%0Uims+z4_U;tj)GupwSanf+wLXDwQ6R8VH(bqnQTL9zou{Fp1gtk zK;fr&BV~O9T^#jF%O%YeUhlGidJrq>CY2;x7d57)nHIqTer2s5z0l%chf!LPul=PH z9c|JqAtKTH-Ce!u4UtgYbw@#tdN&by&ERPVCBY5l#>A#H>Qb#9?BszFW2O7;m~lH@ zLToi(C%<61a#Un;za(2EpNrD2f0 zV4cQ+&a!8aS;0(-2VJFSP)VVbIj#m&lCuI$H^pj-As;SNp3JvMLegTAXqc&Vj9&3|kyW=$U1g3CUvzIIs%ITcGfAXGX`S}Wx(e1F8^CNjC8%SY^z zxZ?HRaMqHugDZH+Nn6lmQAU0ucvRbQ@=_V{VRDIQt~WhO2*7$`X#V{J`t_hwxd zdoH3H4Fd}nmX8W{Fa%bkH?Sd%8bzb>P)`Jd4u59AqqId)zzhsG0HB|Jy z5Jo~i!9n?A)EKnJc>P$C_mwAV*2KUc<}H1$`=S{G=JTF4|3on;;7~jd+5qdxzuF`g zD9rTJ*X>~&NWeE9DZa&~DLry$#U(joZo@@dYryOq{Dx~FiUyNl)H9BP*U;4e1fL?K zBZs$SV>DJ7)ufWMQB`Wv4&CFR5k#3J6X-2DL;%+UlB<70nR58mD^;g+kovi6T|0Bl zt26eMy?;c$%`i$=^sOOzmx8o$art@I2x|{3MjYJA_DD^MPkVzm8LUcPiP?Pj00ARc zrO!&!8?wdwHBlb$(Ph5W6nS-${qAHT<))68tk%G~wghJVb68x?pNKzX5GYob^-lDH zkGZ1^mTce$gG?9U;i1hSvuEzlv4iSAj#V|#OWwx2A8+kwf6!wx^34I&m~Ue4)CB6! z>RcLwQo2Q~HJSr3i~~3v>;HujH*8@Hsypb<&jA)qE}Htvfh$66c26YgZTM%EvBnw5 zazoSDEWBp6JJhdB2txCL4c6vsc&r)p($jzM7(^m;Wqrg4Ul%JuX9Ikbl}Ix5%!BYp zE#!~L8U24!0?dkel_5Hc~x_H9x{W)2{XOCLiU%2S)`FZwiRR3=3 z()ND8as2vvEA9QV`?!Dqyx)k+8d^J9+P`>ejOb?VV~?sFLI-z4OO28f+Py${K3O@v zf1k+JtL^21|AFPJ|H*EC&(;fLRvn0#`SJa_``}wh^ZVQy1-L=`qkUxRz=*CN*bbof zv4cP1ZAU!)Qfke5$ufwA1X~^|EmV^~D{ooGvWSHRTOBGZME_@ED_gL3xjt=3B)TAZ+Gk?OFt(vF>v|)qWu; z13DIzeMuBUpPGt-7z4u4#M}~uW5EgpY!xuV^&Ei4Osj2hjh^o0cs?8cY7hqp|SiIVzanJmy3NQa^ z?g?tkM^S~sq%Bn^QXefB1)J!uO4Y7{>Gs>qkYWoqlUDavon}~j7Ue6Si>6_lW`Tw( zRb!16yT>Z2t@xgmpzWBsim~lU$VQoG6qe`A@0bKwQx1>vaJRc2+~rMqbV4E8F@E!!^0)Oj2wFo7cKb)25QNqpUhMonGiFY*6vL%%E`;;N@1vxQ!7YtUhs{Si;v^Bz2X8OD3ScE!S6n_WRE z)JEu`GBi54p~>zqjxh;AMoWmSP{TzW(!bTdjPOHc*vi3%%D{?$>pZnkBNZOPxeyu^ zU)mvu%04##R{C5EGM@i>$;WWX{b3rE4vGYK_>{rni##BA_-y3(7Qy*Z!P(g!=es_{ zf4PTO5-WAw%p7m~8CdN2!Y}bWLzQFB{56$q+BILNO@~j$o0M*N%(%Hc$HTQDU?>hG zL5N8bh7ymYC}1fLY(a=y6vhL?RD6-e;fpdN_q=Myc0@8E?GC34#7OMict^AnS~F|g zL69NJd$Zn7Jrxo258=bRXBhoG{PwEwn#B9OKw(xiQamTi_Qi6Zk~IoC#&RxkKK-v# zItU{rNzh?DkfI=!IIIN$ZBfu;e3-y7%k)TGf9_>{t|+GlA6<+q(M_>Kb*XFubfQHQ z3APGtJ-1D)1i}C1r|pgo+E`ayA!(wyvk+}U1-fdA$gYh7Vf=uFYlmw(jfs65A z^MYLB@E#b()BE0^6rW>w9~#M>z#}Gebe~=tJR=Yzf5#aK3!ElO>rg}*P!tc7Ct$Yd?r$P{B(x>Kwmbt#*qQy43Om?YpR4oFFeS`^9> zk2WvhIU1yn`a>3rKXmsu#`A5dl?GzP4qb_vcG~A}!zd*4r{dA}1dJsAb<9$C3q)!& zu~Jbwybn2aU-s2H(UXy+8K{55D#N24kCOO*bVTW^dWbaPpy6=CHa^zAy{FrWk%9UZ zIyns*A%Xjk&u?^WAWd@79~16nVHz5((jOWG9jiM1r?;*6S@qi*)EHMCP~)t7k5aq@ zl$wC<-(~oJS(a)0rANlpSQmdHhoQcB)mSz?8qE44!3>eap~eU}i~N)0p%jI%#G%g! zx>5LO$+hy;JD0%UjyELjP~`;fE&7l`pcCbyVkHog{?GSMKO{FXCj~dKIPL>o|7)r1 z#kDK(Cm(2MqHT6U@tWqXI?z|iC@lRSnZ=!Q=X7Oo$A?!xnynT*9KH8(049GUsC}K*hM)MaRqZ> zzu6>b+ZWSMnmuY4bdttH@g}#0Vqq__f1De z2@aagr8n_KGL>B1Ryn;VeltbH>AcAdFw7Q@3Tx3|zmL!4GB8e=G$HsXL*#hD4aC!f zssZ+_9SBktMPs*A(T8kI%{wk<=84 zU=TZy(#~pmJRj?r#g{G!P}lh=Va{8au-k}YmJa$#$14;|Rd4^9>{g8$DBm7XvU%># z90S0Go9Q*9?n|E`9&V@SOamk>*}=DEtSl8ly<61-)rQhzpwy@!lW6BGz2T1+WL` zh55Za^FjZ(-RzN_cJYS(kFM}+gB!f@Z@P7k;9px*{+(`hHaD>`q5pUJ?@a5ZhMW_2 zD{8M>$qrznn^+OWeB!XPXaNa#Oo}%~fS$5;U0#VTHlGPh}E< zYB3RY#Tvt#zPGa(H=*PLRiU=BbvjMDq($Tr<$ZbM2Co7>>RSJ zZDqQP){;uMn$X0o)xm)y)23#<%%|3Ueopy>XLNiih+>Gw6)>HlbqD=;=0sFia(?j(D4w{OM->{56S&eBsRU5x;ASP_4d#2W6 zZ(4*ORwmXaSYU&_$NCZy(g+Ft{J3ZpL!-Sl?R`tofAcDekE41gHV~T+8Uk7kw!~LD zwYlVm1F$Edd>~LhhJ{SeIBI4wp@^kZl6#hj)1n}FSr3n=R*w!FHUn(Os+jdsvgctG zzpER!w4bbGS7THOnyGKG0Y63iXTdFZURY-3a;mRbx?DAcC|wj-j#dShH!Fc**T}(Q zZ7hM*2>BECQNibZvv&l+D^4>tUbm3hO&W&!7LqA)PNg>{WVDiEd<%D&7Y(anu}7|& zW2^X`Rd^I$Vbu5PH)Uar;oDwh?qOYgAyKo2JkCp!*{by?sQKF8M{2_*Wq+IU5mHu% zjiFeYx6xYFbuc|9-eX$x6l-d>`k62s^wU?jBfCx5o~%QHd*r1`;>)LKt^84&26YcX z+E?#GESutFm{?(yOb(}fY$kR!J{41># zWztR?^*9L?s)d_@I30Yogt;=Nw*T-FsjwF?jAKLABU>rWRwkS+!Fms4q#Ew^upHbe z>O3~d>#JOUv^T%})%Mem4fTOHKqf*rbi<@hhK7>yj!QatvE83W`9#E%#9#%+3EUmp z`l;W_o@%vf*l&z(vNxt3#YA6SD5YF33Nr!JJI4v)KhW~&lTiY-x6b18;gIN$`5p|4v36t=;3EA6F&B2trR>AcOA;ab1xzK z+I?cr3*~CWLw*`_axqlxTX{34mCbP07q*|J)v9LTlo9N&1T>~2DPL~1(s(V%-$v?3 zknT^oQas@j@oc=Z{v2zf7t*51q%^TxJq}o)8#}RQEeGG=W2GZjnvQ62=!5$cYbO`9 zEn0i})joWYG^`+3mf={QizrOy7(7f5am|B=kQna()5yo0c+TZmISg>;$PqvE`NG$; zWU71W$4gs)M8RwINMt^3RiK;02m<#T1O|+SDAZ?_^Oq#{* z{-8{Yu!a8A6JXKvm{~dgqw$*p2b3%oaaSjJ57oMP3@ZJQxP|FWdaM>cOUcl95lrBo z6x%shd$;ZvR>jh5<`!F&S&2lQKr)6$soZXjwO#Xo2h%3kdBVG>5G;wXxV9!6j*i8) z*2Y?laJ?j$HYet|7_Kcv#h92qK{jTaKrvG=vdr17>4DgtSaNmHLBIAkYXEgIV<9q2kGJ<*L zLL7IZ9H|$NBU1Q5`5YcXJD*@6&E6CH*`j&tL0yeS3~al9{yFn2jU4*v`ghxmb}#_I z-vsc#KB+o6dsv$|{VUyjiEH&Y;dbX0@C_~q_?}-Tt(Z(PSMSqILJb+O8<-RHEaVR? zS7818DldL6E1N+7#rvi5MVhVgtI7NFo-lF#Xj*o9KJj;U_R93+G@YF;uC>&wzI)FZ zS1a(FBo;(yqbqyKgERN*nFH%@SP;J1>=@hjSO;!*#?QXWPQLHGVR&7dgQ-RX5Fmg1 z&y7AOgv%O{IXyn?mTSFs?tuVmt(C?nfF4h|g5C!hpMwLT7SdP}2~?XcXE4HGD=!_; z@M^D{m(IQQW;aj$9%fo*cfvZb!0w*!>oEu3uWZmVyuUOq!3abLL^9y1U|81B=9>ih=%TMrH!lL?9Onhl$gTS;>*5G2Y zx~9IZn8sE$*Ij5Uw;K=we2F9k-8Wm{m~(f6U~AnrY=X~?z1)!6@B1aS`4o~bny#-M zzwZy%*QXhg826K0-?h%Zm4VxtO0_XyyD&c=Zk%v8cEjMot*K>~*ft6%7&HyLvt=C>Cd}nA)5|9R0+7)CV1Q~w$VKlMUnA=#dnMGRb_45{s?%)_=Yk~%x zEt4z)4Uk*}-`wslA??)z!nS3+xbND)yCP2_d!^)yic(n*Wx;OKd3o9H&veu7(d>B} z`t60zjXq0K2_+c|GPo2}%5zDrGUN#PM%--LgfIuM#|`F3Hmub{0o`e}?z+m6nq<`P z7WN7xR~MS=xn#z(7%9sro%%!i9B^)x3OKT&E(x1Cpo~w67Li7IqPc}_XR!;(iVc?< zt{iq^t7s=pvjyqx6w5hC5f7?_Z9463G#C0>#4Xt;+=i~nv?CuWU_9S-Z@_dQlCA8P91NT^EU=QwF z2#h!yzx$y#XG{-sVElE!;*sg{ZK^}m8!|+c@J=>!X!7gjrwA`)*n&~lodNDV42)6) zIx?&>S!3tcYzxDYJb1`bbG%^jyAh1(0ALjb(xnS)(zsKm!m}MCd8GleGqRTZ5l`cD zJtkC&pKsm{e7#~rHv^+IMDRtjdV}8Y{<#-61PNJ(=d;FPHg4>x^|1O7Id6~Ae!y;F zz>YxxOxbSi*h+qmxLblrn0x|!keK@7&)aZ;X5!1HjuyiEBEk3p^t~DS0`U~R=nOH4 zw7r}-@&k9qC$SBLCO7n$D$mai0VX%S@%7m>Y*z`K0>$zSne+6^iIV($@HuKkprkW0 zv`}I!nhhGoyv_`ZIUp54bJ&LS#p7nLo{*WF)f$lC+G>-sB~mQaL1}etG#9Y4CqJ8H zHT@82?{Hd{GVrMZj4*>oyq^&Dx{r*`(EaobXv;zklirfEZrm3~Pusp`?$iw&((eS- z7|M9PQxQT=pT2e(alB+?r;Mrn zjTafcK&TF=NZpV~ym&3v6#^ir2M|&fVmJZmA;Ly3Enrxf+dv&`+1;Hl;BEc*=~lud zU8<4er{Uv=lV!W-jwJqdsb0Db0JL;q=#mA5c$d-zoSJZ3!;A7WL4d-aj9_^@CwYJ{ zzYXwaj87?u6X(%h`y0FTX*}x(rbd_0U6a=;g}SawXrC{=$nFZN?nLBI=O1t8Z`)aX zV<=`2s@$l_u+d17_~h$0pS-kz2(I%+omxecTM7p}WwnjR8_V+5w11LMa@mm=O15G$ zGZ<~%U^D2H=zwo~QFELiwuVMMylq&zOMs~HoX7XLbMNBLWd=^0Xr>1nX4$^55@K@) z%rl{i$Q29uH94%Mh%IBmnjf4`OWQFrw#QH&G(&# z67CSMZmCG;)Eg%w#vbzPZ|4@=;)VkjxuMnT;f8MccGo49IVZD~Pt93P^g1x%W2nt- zO%@x@u@kY#2$6`JB&M0FL>HBlV{ZUQSTyT7p5jPGZkN*BQ^AwPA*K~Df3dhYsIP|% zJtWLYRGynyp)Y9qulpn(7>og#8*=2}83Ur+Di!I3VAg9lWgv^t8D(aoNycEM{HA7W zcHknjS?C9G=c&wQib^pY9Ya=03#}PlVRqC(tds(4800%KQ_h)`LwO73xeyPo&N}?aiCSQ=Tj=;`USKmpaM_a+fC<|02 zj7JY*bUaT*pP|gI2iH=(W@wdEb+VQq)Eb@tCSI^Je3AP`tPb8Kp~ zd&VCVP*8%?bm;RpnK6_(J#_ohb>{Og+K8hDqx>BZHr_iNQmCGlr~u&z)P{KV+MR6P?GM$0 zpAd)8njAnr_7==F5;5az=_!HpK9P#9RtTK;GbmRwgv3%r^| z1?=r?5G26VAB``MILzlf6F?LuTs5n%f>g;VRz;L|WN?$b| zEjxRVOFl=u164nX?=2QPTt;h`V4tAm;|LU1o>`zo)!#7~SUq#`GQ?eX@YOd4cFluM z0hhjbvdDpqiXo|1SYHye6v3ugyr!NbSrl7|2}^aRy$9z9=MwC`^Df7&xR36q)9(r>%$DdpHR z3FX^=;552LIwJ0O9X@%lPg&Xn6QGjO&wFKk|JbmqHx?TjGc>#kk_zc(gmz3@Kd zK3fX>bi7t~R0qvhp=sCsND^{J}#{! zW`CK-BAff1Hn@YHxS0=kL(NlM#}4etMAW@nZCm$A<3ojno0C3R;W})ld(yMYQX^wT zK1z}M)OEc75+p7Ms28MO$l~%%ie_y!=Z|zUv(1JJ-x}kTb&ljqZIbLUuB`A|YdipL zbr#|tXl4o4DMkdd&(mD-4^_2Mg40~wNvhd&UbI>;#bKxBeYhHpwUN>F)g_Z+x_v}i z!agdzcqC35BZCl9X;fwD1=9ijt|Gh<_#$+ea$6)a{ufWq0KEdxv?3f5P-+RV_ZbeS zI%D}7@QjO*^;0)04;i{0_zp%aU5xrPiOdB0?wF&xe^?Z}(5IMWh?ph)xvdNDi28}*27iFQR=PihiM#h`8B=*oqxE|q+= zZi(;Yr{LlfOyNrz9k39#-xCCi2O=u*Z=@9p*(wr%>a?nTdlmZL6^)jIDNnyG`PcBk zy%Q|?Imu2UfK)^5&dSE9C@=@k3Vuxfe9NL`$t(g#sryS|{(ps6m9m5*0UkP;rT z-a)Y4COSs3c``<^awYtN-X?IapSP!Bg$PVyJf9TVXe?!XNCh@sX(hM zkNA=A^U-I9q_UOMN9rPA-egyZl3?u)&y*Fm$x?IDwcHmt#Dy4t-E^WvgbTu@i?{$|F65K0-v5P!3!(Sh$I>{-cgpz`hLnNUcua;_3 zOM^zkDNp!mBw(}`Dp&X3mvcUsN!v}rV`3l6ipS>niqyZWohUkfL1|TFt$C1Q*s*8M z!Zcv=F~_3H_X-jPO|#bb9rYoQL3RAQfM&NeymZyC-qe7viaK++Bd|ez5kC2{4CcHH z>l1APTS^Lr0)#5nT|)7X7;Rw5AGtf!MbYP!eP0BuC}?+BMg-=EgsZEAa!PH)a3$5T zjd8`PB+zf7yvHqxK%&McquxU&|z=nds+V-3^m{x-qHn z%=f9yEZg(WC83q6i{V4=iJ@OnJ0G{(jIY#}yvWa~za~r9?+sY7rin)6dqS*LaRU2TH2qrcecRPtlU{MJGU~e zgdW}8p(xfoOtgrE(=*E=T4Z%Y!T4)@ps()>4)v1ny_~E&apm>;jq&y<^JETUU7lZ% zHLz+>B=BNYeAs@Mj~lmTjusYkvtYeu|2lI$IsgEa|1jw02974iO3scJwr2m! zr)zFFZVn^*$fL{NQ#EP5-%BY@y1Q%h-I+FBwYoHDbZcdT9ZpK+NWE1tv_&Ji zY+7yOR2dyHian#~Y3S%P=ployD&JPIgn3pb3@(xw&i#puxW6)$ETU7D{)OMy91a@(hhsS@s!Z+o>j-la(N);g&p+wZ@&l`x{`)i>F^(dFi@#nP;->YocQ?2r$hrz;I#`5#D2{p1IB zdgO~6n9w_5KK!q6x9{CZtJ~7M-i`0ij}DhglhpV!y56nt&mYDrdvm+q&kmPnO3WOL z9Ty7Qe)oJhzF(IXcIN($bi*N8rK6Xv(3U6-u2e2q@ghX7ZC$rIk_|KJ{w#<8^;Hhf z7T_di@10YEQB8>`SbSG_>shF6_#_#zH5`s{CaeE$6S-&-8Jy9rpnI-e+zpx@f$Wl# zJtJa^)YT(%+P<2JW1p9dTRmr`xxgKYt!thlG%SJjf&;7|yGHxNs(o+oML02;d1dan zw_=8lZb%$={5x$qidWIx9z0+lT5SwOA2Te^R0_|JsC(QWVmxBc>W9jWC`C6Cxhl&Q zCWi zrTDd)rXuY%^MOqpbL+d-?^9`0-(J;ng&*H0pC&}D#oo!l5X z|C3KmEv3E<^Tkx%Rg1}W!6zAg;O+~uC}Frsp0oMg@GOXxp4Jdy z1ia{L-$ZCcSI`k8`mfei zg^rfgkm;AbeNr9$YqdSWd?}-|{lJ~mwCy6tAMUW|L7*X@YoB|w@6AW5DL`2ygur46 z8F4ar$!Z=Et5NFhiMB|a?d%cdA$PBiebzJhOOjckwAIIuI;uh*&3N^D$#fUsU}TF@ zfSB^pe7(=(SZ=;iZD8cnW`Te;l$$gOLsSE1L&$l_)b}uMg6p6L)#;%R9E*1WyQ#Uk zQJet9eHpEg*e@sv^{ZmL{)+;3gPi=aqq4AK5z#OF-6A3{vwi zf4K(YtfoC^)44?wR#IVgP{}>Kp=e!rn+^&wrAx*&T#$PeovL`s)z~44$5kI*?jP9u$+2#se11}#^+^Hk?EH`k2AX+C5!n(n;Udbe2Ee@H*K`KGX;PN))Ai(%}7^kD?z7+rg3;Hkzs^2H-@jIvhv_hTBB^U@; zviIFUIfWgNLV-e(H)RzxixyuoQj>xzUl2gOlmH|czKp9ve5x>-(Uj-k`lQ20QwKF1 z4tVu}sUa|?toQ+2ei%6{GD#Q^f+?7fe!@%4V>s=x3Xbhh9u4`U%e(l{+5PkGToC*! zyB5VH0pxuMU#{MB@}%=oT$$3UgdvNo1Sj=-`@-7SsJ(_W__UJ16dS>8<7tEcoD(-0 z!VLy~k0IV7#59YfRO>>$su3u3)ZR3u;4I0PE$b`_6w)|j$vq1tpW31phD^0QEh-(2!u_``O=rV|AB%gxbF=g^UPkAvBEX>d*An~w3^^eB6hb{`DA)XWp zhMa%u#?LqeljR3k%duLUTQB!XNJya#tjQ`%w9$7t5VA1i*SCOJ$+%N7__rh~!j!Wc z@$Op7xUd5z3d`gGGmEcuFmGf8sDi#T@9?HTHmk>IKi}a%dMhz(F$ht9kzlTQVPHyW ziz^t!-0?@BBc-s5P>qYBOy|Q{P&%qy{MHs&?f^a8mKLvX;F^k;T^3iaOdZ6=Hsa1W zK?W53;|K39rVd`NmKW}O--tpe2Jo{TRKIq@V2S2oRg9Yp{l|U!vp84(XTf64$z;SCjF@f`B3p&sBgL zY8d2BV^G&LkI_w zajqV3=?K2ge4>^BPxrF`2?bje^#Dv7XEZ^F={6swTvw(>5tV`zheM>Fj$@1q^b#5SlbT~$QZ$5&1n@cVYvwy=I7UWa{g@s> zmR$LzMmKx193JeY;(<4#BmV4DiP>mRfe)i)QrR=)8uYDUWro1fq(B!g^t7VRbS87e z*Mc?-WHBq5FFsiLRe?Z(uA~N9m_XVl3}?rk&(}GPzECq3Hv^M`2r(B%!?^x5HxyCI z--qU1St^A<9CsAwz60*72#Ayh0**L~SbANlLu{#e=F>L6wo2(2uk0Vvz6gSCL(NxPIkQpY#iU@Naq~!HK}Y7@ zx8SCn3gLcCE`66&LZD_ zOlC2E&d^5|dX_*>{PiG4JUQ!>sGqK4!!#bstIHjRNP-_&pY3yN$1pWUvZie0rkAOR z4Ud`@GMhP3eEwREFskfK*;;pHxan~whA~0Wg1`VGw7M&AJJA=oDoLGKUPKjVRwLR7 z+$diidG!@>c_HKR)P%nYz4Z0+O^~WE)065*wRdW6q~Vvlz}X%#M^WU3fMYw}0o!^X zJh88DxJ~Tk2^1K(`SZ1v#7f>!SHsv0C*HBHBthX*?ct9@mz2Tsv(REu_=7}$h*ler zy;w{i-Ldx6VPDA#u$Z|387~fc$_T>RFQDNfg_wfhO=)gv+&B30wNYk0r<(*n{vg_h z$`KJ#TqtN=+tdbj8%+u^wVxOfBVkDvyXu3(2Dolcp(TCxFSAvjOq-I8WFVvLLxf_0 zMl^9iBXc%N$S~=avT-&xAQTH%{Qk3kuC;*8<7iiFTA(9mK6?Ylie~(&Y0bj+P21$s z2QuV?lemr=AD}5x+O4`?m=Z9NwF?_Q3h1C?ivsO)t&?DD_^u}7ESX32fkLc#Vab2jra`jMf{sXX4Zqh098B5 z2iP$l0(?DI!aN6|qSdepo&wd;pKFnW)&Kg>9Uga9Hf{n0$`vR6OC(ogW!PzqF*l-X z@O@|(1s8NbdK}x`8;>A|^zbk>DrfgE3ReDd%=MOsvssN&Cj)*bWLKHSkriQ8DJh$x zvK3D#xM|IGE(^TN7N%n~x0a!qvqL5vZYdzxtz7{jk}0IYBpfCYbi8ec%Ws_PjV)Ch zCr9g7Q}PoWAdul*aGDkU7*)hKR)D%{94HhEhu}z3Zd;yZ-C~xv-!fa63A33wzyuN_be3?M z!J{2RQTQQSc-t%&yMi!RS=!&g+~~?$e$m*R%9FKs0qoq;#oHzQQu?Js!yggSQJ>P7D8k?Y(ta zRnPi9jD#qlG}5hrNF&`KjdXW6o9>YAZlpm#B%~XprMsmYflYV(HmDyJJ>PTA``>$A zXM1ffaLwHF%suzanpyL#XYia^93%Kt#?r%&QnSpRF%Kv1VKW$W!vN=?u|*{{qYGLa z5yO2DA!1r`vlx$OI((oW`%ztk)_H%v_3Cx}&{`R^^h+=V149F?0CfP`5|#jSE07zO zjj5KIrlAR?ovB`T62O2+-?d%8O%3;d{yS;(dNwPv#OH ztD8n8JoV}5{F2>~Ub`?)J!DnEEH>qE#9)Q4k8Qsjc$^fjzv;A}%(!33xl=1pdwq05 z#n*kJ2tX@@y_YSq>?vNC0!b&)@YKq=y}`|Gum+$Rat#|WllakjA|y< ze{b;v;Bs?xdg8hY?Q+7+zI8rjWhip}{uBJt;q#UG+BCPuu7yK4 zUA5+yaO$I%s99EubjKS8>zN~8p7CbW8tc4`Z5FS^atEI3m+y^na*nmhahz|KEXwr? zs`o)B#>}wPGb%sn#1?6M1*DOT^$<_ptX;!C)n7?E+_OJkJT0~Z&|k)mQ&Jj^B_*@c z70s@&mBjY+4YBKtb|%xfn2sjkSk<2_1n$Q4gifX$!o(e&z{Nf%q!bQx9%$WV^PwjJ z0#Em~7gkf;IJ^a6DyYIzNBvV=&(=}2z`zaZkcL?-n6O_H^uFOzBIV9U=Da#tBx{j) z@Pq~vJ#)P!l;g|*CLCpu7L;zc8J;tdr~3`pWFlnQ>x(&0_$-7~E_@1dP_|hoU>#Y@ ziwBuKA`)L)YV3d@L{BcI*B(kF_2=6)(tgYcg331~18ettQ5L`-WG+9=v00b`G_ZTe zLA#g2-r(m~%p~yCf~~u2zVAua@(MwaTo?pDrtanfQ}h7b zaHYXPpKjCPbhm6w?JW;|e$86+rx(FWxrpix%h6RFZ88&vQ=THi}DPNnt6#XuyJH zwiE_x`7knJo^yCAkr=6^zbHj$L7Lcdr2f+LeT4*s-)s5)l%%U*+0C>lIP z5J?T+*m&BBFDtQ&G0bvn2!(qk2!FvkhP=Ahhn&^V;kqnT9>SX&y(A2nNuU&{|&H zlF1+l{e#T%eoOonuABU&+$>z*wNvUH15E>e%iE*Ddp~%iCP@4YrlE4b z0IBaotbwQXPXqEHwI1Hq5GxbnM}sRHh~QJY|B>{YIzX-Y8*2WWO7I^BI751#IWIIH zBe|UT5`oZcW+y}rywTpEgVggOcFoh;>wCr5U@0JJAa5Fhy9x1!*;J*s-Djjn^9&y+?K_!gJYDW_5I6Y;pw}OKZ^VHoO1Ld-x4dx(3jE-$z8?l3 z?UuJ6sGvS~;4KGP^x)QP_`zQcK=ONf%bQDB$dAtAlJ>2$fTa0bQeA}D+t&OIwH$l_ z=7#}0h;SSG{_wLRJ9hCQH|mAZr=men0@DJ!DOU?@kmh^EweY`Ag|h49wwdxF0hAdDY#%NraQ3;%b$!FiQKE+GAxG=^K{ z{vW85Fn;7e3}_bkK4>(aySUl;QdL4 zhxLzP6CLIsG7vJ6e~=;Z4;eoANk4j|&rdSGqu!8#@(&pVg1;E>4;kDnKL&0%?|;bn zE$KgG{EGSy86X2b-I8IX*7RlTT)nmj?fR;B#uZcbEB%I(-a<=?4{K zL%4v0gnC%UDq!sXcuko6B{ZPo&TCA(wonzM|~J}4{2Ng9mHB-op1(l z8(dGbfROQgiT9pD26MA#AkDg6U%tOY_n;*|mWvcVTgsrA*eJD&8*WpSpv9-n(U;Xa zs}B}xfg`oDe8TuX)7VsWR+k3!3cs-2HqmO`e5P>=Y5V-?B8_S+ zNtp$16gH^1r4Y*@Ij0KyV*SBmK2gFIE=90l?xb!j+}#m*^`z<%*wiI%Z3(KJ%eaE{ z$jLb9WpGfYT4B+#TA>!$R1_|433Uno_47y15Mfhac(kQb=Cpt&Q?5pwlpa!`3e@Ew z4T(#FtR*CM-n-{nZBHrr!NJxf@mdAaX>(kZmEEpT$Y#IpIjxCosqwZNgyEx!vgh1e z2{l5>ytetm5;ZGkgQFkebCHG%s@|5q)EUc#EHmDIX8KREjL0-Qf@t8hC8MG7=K|S> zMq^*ZCNYbRWj!We+C`T8Y$`>k+V)Im<`dM&J8IeHC=Z;=W5-hBHdjT4s8^QL^X1tEx%Z?^ZRK!o2`4E8jW;?_(hztyKb z->S&N3H?388Eh)eEm&F#bntEZMdY?0MXG~+hq!#zkABNYfBSy&_cB9QrGzA|SC+OP zu{oJSh&$F$-e#YA0FHohoAwc#4Y2wfeGp^s3q)?oW$;k^emNK)0`u1Ba>Q=;w%e+i zLMnbQCs+fQ{{AiVM4$h;_`_5z?td;;g8jg_k;YByF?js2^%0xh{*UtI^9ygv-!8PO zn`LY4*e{sJ9#8=u*SUoOz>)>UsvzVy9o(onLJ!*H%T6EhSSEdOZun#fuh!1#6r3#Y zee$JmKTEuNyV($|$j9^>sRlqF6Zp_K;# zp+$J}S6iItPf=&FlpG%kwXQJ5wUkZKIXHPrPes>%BHfX!_>4ey$D=jHqbrLn4=cNUvH7|IM(>rXPjQ#VuF?-kD8DXX_TqNbAG5|O^{|JW z*Ec-0$(Zsn3Ay9a40z$+k$x8XX=E8^uokJAMq~Wav=jp@%^vUBByE1b`upIE&?=t+D z7YH*fT@!4Ea_vkv6Ifki@g;Dld|PDZdt%Mn`3Gkj8&hFZE zoLJMAh(6JGV9I2KKZZZvsB7zCnl!O-V%uPHQ;VU8t7nvsk2FzBIKFb^JF~kci%8hj z`?594S}_5uYTd2Ln#)*`-m6KMD^L@hG(FZe8hj|K!?9b?qnb1}6_&RyijCChCE`odbg2)c9JyhWdak-v6yAEMQ^o4)qn}!(R0bd zD|&dB*Abe-)A=O*5%?^6Kwn*1DWAG>8E|- z-AH<>TDuX0sMmUV1UY8^K9kr6&iNrcdW-%OiHkwzCQwkYa6ZYB>jSbzSH~ z8Ig-7zl`(C&xT;0Z^>o1U*eCD=Vt6NDA)P5Tbvo__kL`5IeW?6^m;YYExha!xvf{7 zXN%6NJYoE0afQTgg%W8ZC|H5f+i=NQp2OMTD_}?MR$2V#%r;QSe+ZSux%ymZ5MR?X9y`pP;EN zIrG`t(^4;i>chODl87YZgg`cKIY;fN=s24zG5+2ubY{1%Jmg^7c3ZgS%4vVeY-QV3 zesucrsr+XFmf1G2tS+q8(=0)@U(L18Dd3iMO8a@78SAU94uub_NeZrC4YMr8&CYU~ z3`(~4B%QtPMiiohc{17LJ`h!LB0@=>M-AUVG2wYm{l27pwkjq6D~&+YI=dDJozoyC z;5hevS9U04K_e5Cw0Gng0GnG1s%LW0hC>y4$TXdVnuyH;+g2n~B58oZ4YLW(+tR{GHYEBB@M=U%mt$5Z$IhFm z{!0L;T|uGVd6I$gY9V>AZksPom7O%FRM+FP9m{)5YE8@8h7FNUp0U~dmpgOCDM6f0 zJHRz(LfI>8ygkRxilXi^?}ptg=>p=2-nw>qwUG>c>Y^s0Vm(BC*N=s4!0W%pTb+3k zJv;$=3^@Y2O7zVkF)zGPxP=SK%Rq%{SvBI30$@q^M0v2I6v| zS`61PBPuC;{FGKxH_@=LnMoNlm zvNIxtxUa%nZqIce6MqJfN&(d5Ucc=&S-}%3apXV#>x0?}L-VZvh?(11N`GT!>knq) z|IUmzFCi@J^c$YN_we*e_MGgq>~FNbGZQx7KRvUlZec4GYY!7I`wl@lj&Co6W|Jf+ z#HLg5C5Dv$>kS3|$Ry4hM6oIQ@EEtZXb&4t>x z7Q)x+LB4<=bsK`JsD&R9+X|e8u-ixrY6_~Yz=smMugH|F&7lSr7Fa33xJy)}O5YOO zz_YwMfs?4O*x7x%s~I+8^EPiSW836I7KJ9Kbm2)#TJI51vBJ9_pX0g4oZj;UxzJ8r z^Auf^UBx3;p6)fsP*bU4>XP6(;i)SDb|GDaLoWw;xz-0Ui+#^Z9XFpfS}D1lf!oSf z_~Rfh>hd)JYxL~{>~s=Qbhp4|H6&Og17hCs(h4`PqXhCOzOO~?5MG{;C5!u-IPRbp zAU1;4XdR`S0a+v~Rr!UiStC)WLnDG9PlO^iuC)ZwhaU?oJ{q6an;R5P)&?R}r0{V^ zRd|)Gw@53qtckhF{++0_()3VYWy)5!R zLBS{!+lMtQUkbN=CUti9*4z1!N$#soQBwyK4t$0v8#Uu&r?DtEl-f1#fza)4Mvx%> z-1R8IX5G4p@reNX-t2mSUQE-rv9bo3-t2woT3S2YDnwZSqfWnGzrMLHsn!EMa~)8^ zzhIerH{frAj#2mw^052Q#}iLlt{q9H1wnU!;{1De#9j zVvsJt%D{$8YEbJ?Yk=M3N7E~3g##=lUNe5AqcNwgUFcjD)YQ<*#b#G5JHv)?=r)Q*@+3tspD zp*(6bm0qbA(F)E&>BGLsl_eqsnE4S4I2^q8otdiO8aK;>*xx z*|B^e=Q7*DWi3%*d##bw>GQWfQW#H{BYl(K&+fZT)y!TkT)&&)pLoB%N}uJJe_aK* z>{VMl7tz-}r3jsI;$YpD9@|Lrei2A@+Qk94sBV_>v}H6FljRV9ksK&>FK_aR*%O=U&r~0iTGtZty!vb94cZ``y$8Jx+1n7gX!lXR>tyd(^Jf=!B}p& z4Z~g-O8bD!Ykg;{=Y90o_NNg~g$GDpzhA8n4Wyb{Ti6>t2A)T+VslKtkc3;YU$11* zUJe1&l5=h=Iv!>p&vaSAx(Z>JQcv}_5k9lG9*p`(B6oTzulUr?$|{jfH-EQMcxUn; zVTo$gd#GWX(@DdVH6tQ|Cw;gjh;Uk!f8F?teGt{;6HIxk#N*haM^Sr*y-8^r<6#@8FfZ0R;3%?*GA$aE#ltMgHu3 zt_aA5|7YjB?H_ykI~vrQ=DO6nqDoNwLyIe78jb^ zJ<2gz57(Bs)&_e-07rN)sS#NplkMF^ZfsBB3aPK2l$aBkMI;G11yrn}{qQJ_yC}nX z3&C?X+nqAQvPK2fLG(>e8Fd==8rG3vW|fj2T_J)Cd|Uwc7zv)olWUcBSgsRRvl)<; zf0CA`|K!a+=!!V#Mu+}KMQ^uk|Et>SgmD{tav^vFy|17rY6=qMv?O>jiI$X zS}NM()_Mo$9LLGS08|=BCc1Ib7ou^i26$wJj9d&nqX`_{7Q zILMc_0}!D~VQFLn!yRUXgd8D1kTzndP(vTcS%h6Zw?Gf_K|9oHEb1OA4Rd^G?^G0N zFv`&|R!H4nr}BYf<|0EOgB|H{qCsf{|FR}OauWdS{eZ>X&MKT`J~t0T^aX_C;WTmr zXk7jap?x+%!{{(zVAk}1Q_k%q>OWP^rn0it6a$(oQ1lu?;9PV+_bcD}cudLHiOh~+ zqZ$AX0whI&5Dx*ja&qt0F)moGbd+G{w2)o7g8|<*t;kqDu1uga4}qYzN93!M*$U^*O& z)0iH)i#p3XI5jAWA9$L5!c$EV%O2I^xTdbFsxR4myS!qShPA(}WoyN#`~_WXQXR%{ zYvscuA@faO{CKyiC;}0T6yaWaWjwyTSOm1_F@BDH;<`I5Bd^tT(!@-6@{|t;$+nE0 zNjrUQmFYw;iWhk6s}7iwL!I_<*LQ~g5+atUXG~cS564ln()ii?+WF`--zo{BTO#r* zy+UTrKnT_>4dp(IJ0(W!F2o3QA<-p{uf4vivGZ+F6?%{2vmnrPl1A+0rPpiX*Agj;wg zqyucDnwnaHS{^wjr{y%pWCMpwlx-~eX4t{Ct=qcmJ7@f3M-Nq(buFv3$!AA0%I#Sj z_By#rK4!`*gG!Y{4wOWj@QE<-$?PduxfKT=jiy#aZLEi6SC34DUIebec2S0O!5`D> z3CrNh6dsmdR8Ae##ZavYaaA_J+#8MT3}1zPBL;-9oU=%!rrCSxoho@zu>YAIh#b(L zp`m7S^neNll6fjoOtkjU>uop5!r7X8OW&MC3avc&)P6=EdNxg*Cx_wIBIz!4)W<+{ zmF%Fa$XeGl*OmxLLKXfx_fO@o#ECJ>j|7m|i>F-@6N zL|WYt>DS^%lAe#Z_kj6}Ffh3h@!CCNn!+a3vftjh51ap@ZeDyrapl#Q_btH8g}h6D zJ|osxsk2vYqN(uoj~QEd4Z#$bj03>&X#JG$A9Io+S8B#H^pb4aZVsYDa(vW|M>wif z*V-it-A0_+m0=jGPmOc8`NH!GRoXmYdwiNz$PUJlkmf+sRcR+AZvZiSpfgm(Qb8O| z;@%1On?$gUu=F?PmpXW<@`{YM#p9<1BB@dopYg#XnRps8SaP7I`y_V~B+;_qPHFK+ zbkHIpjYzxM@Zd^{u^*aOy6YS4hQ zMNem(hU+;N43U@BWAMgfvSl{JCpVh+J&u83uVJ>(_%h=ma7Jsfs=|8>Uj$(p1B$FN zhB-;DHg~~Mx4Qb%P_C@hdorsyoRN6Ak7+FP-bP;{AZqBZN_WN!0 zDG7yfJ#b2+(I*(DPPapKS;Q?Y|LCbV1ulZS7c&%B8*6j;&gN)6ox7ai-OD3VwC%?) z_~Q`e5gRj16GqaBSFfLh9OZ|6=#OFkXwB4Q$n}Mj=kgKcH5`NCQPjH-M-&8EPeVG) z)O3bjH)-Bt58%kaQB!Z!wO#4;R~%C;FmhfRoqVh43SxPtH_!HD>-d=3YMb#wz_KQY z9zO$9koih)}rjLVI8_Z2oH##v6 zcRvj^H-Mu}pWb+$)DN*MSc+Z$YBj@S4~}rgzG>un-;II5`w;b>E{gf21Rx z0n@z=iNc9`az3jO5T-dv!R1sNmElv}Z>DFokqGv|HHz8*xlwQl$?Rb>aEJ*Z!MZ5M zkZd4`oR_>|SyU28bvqE#rW0ixw^0v^h2SmrE9Xpwpr8QR>EVaY zl_7}8McF1AX^jQQTjZMvZ4YCzrV`AIDC<4K)E{#>355erC}N(}ZqN>Mo7V?ZwYqqJ zX#`X>yi`LGszP!pcCbEQtesm=z^bwLK4nM1Od-_{3aKiB-J%;0So-8EeW;aw9|Ajt zt$B~5n;ad_>`@fgC_~v>LXsFxITlny*t6MZ=20Vo0XY^dy80Zj2^sAyX9Sv1G^61= zVvmw1Va{G332j`w)o^9bk;^h|$LN{78t0FS{tIT#oe^DRqs;yPusOZO3s> zaB(cMlwBvBwabjA?omk`+j(s$`saM}F4A<-meLC$j`!Vo&vC9z?OX}lAz^|ZxfG=H zX|ksC8LovXq$NkB>f7^cKYc=5B~`uyH3@`x{b>qM16L$z)5kQ1%(&}FYe}*iz~v> z4JnRwe&Z=3a4Jw3_>gv0u%vP=W^x_lkl8v*Apanya@ZmJcVrM<*>%a&pj2AlP|zR3Eu>UwGWU}x4D{mXH2Q@jhib2Aa&JlY4i-l%fU zrSLgT@AGZBU{uoGTWn5JIZu05j#z|9j|i5k>?Bw}jt#Uws-=BA%?$E|*rAq_HQvXc zGo7bGAe~pk4H8@eZygyFt!cH3Cre8IDz>+iK#eq`zs#v#%xG93B?zNxUDuuW#kVwS zWjl~wm2=0;hJGT-Z;1M_fHj>|OjfRA@%#4ogL`d0&)-jUfyXR`V|9zS5YaTepZpx) z!lT~Su{<{LVPpeAsnU@;0qxmAhM|Klhn(IgsOtj_m!(WA;!3yiO}34Jyc5Mx5}#&V zaky;uWb_axHk!m^b1mklcwg5NtFJ*3fI~1PU!Z`i6hnl|`kcor)I6jp_2|Ts;d3S+ zTddrtXeTEzA|TMGQ5p%ktVug|`|E_WcIMOj8qnn!y7 zV}4-3Rh7T$MpS8|go36C&Vg)r0TP`EZA7t2kjrt0S^HW-M6jv&%H$WTZT7Q{C$TA} z>dqbrtUyfATd1gWQDaF{3~SA*f#fnR#7QxAcK!IP1Iv4dA7F9%%0y|O;mtjBr`QXm zVSy;fvmd%FwF?}BcGU{NjJRT&jZiZDGJna_0AlgzsTnBe$uOH;> zb@?7*IrMvu6|$6~Ct%ln%QDZJIS!C9ln~dlqlq->_^&c)xIgadwC{#hJIi`RvhQNC zJ4UD%EGclF>b__Ioh0_>((VaIlcfeoGB>jMa}7&(6D#RIU5OM6Jr@!wDpvbPgO)iNz48?G>B_sztnoF^Qw9wG)?_YMZ&e{l!`xtwMA(Q2 zy@FyCyMkh(_b`oOW8$RYB;OF4WWbl-EZ6@uygvzV?=$GiSJ3xgtV#bX>s4NN88zP= zMR^9V1q_gxdinYz_9l~$iU@2Kj6)K7-n0f;n#37Uc0E5^Q!h@S?)o6mehyRb83Yy9 zBk+!0rlHnP?dSO92n0%)TTNc~{WNFUI7hU{gBd0f99pIMs3NFI*beX47dubkO_oQs z8Y^~aeGV<2+;}bbN>PW`X;^i}#Zz!7%E-grIH)2?s!?@d3KWcAik3vJF4(`|N$U_j z>*mBXyOd1)=sU|}dSIWc`^s^j`q`uG)C8qV&hd-CveaV}8Ivb%~ZY?ybN>|W^&L1?PnIQHsMNV7fF>@$+8xpp8|9f=o zmN1VVC@*&?V;p?sM5^&Oidac6&(p)ORuaFEtXZ?5k|=srHna#IRl7LX7r);pGlsVp z58h#{*=Go1!K17VU|)}8Zxh>L_TdOGpt%CN&6wp<|`1u0J*K*@QUhXPrtT%qE2(Pukcj=LOD~ znFkglc*|;;qp|4VR4djyTjV!IBT0I#v3+?(tysM_<>o$-L`kskXo3Yz#qJWcOb0A1HHa)2_dS6{>~(3XOC20YZ6vFfm{L`9x= zM(dasa~Y{fcI-s@V8%k_3%7O2jRcC6w$Bak8C7q2ry?Jx29dH!33ixT3U>tM$k=U3C7{h&ic9s;S~-EUy(| z!_XNle|GnE4geC+l!tgq)+c?r*+9_%>VAI?VY%M(H5MQdq=T*^f<3tR&+8A`088C( zT@&(u2g-r^k8^~oNHaCMHxTNYx1&ro!B*X;03v$6^?uXy06PMQ3h$SOLg~uab%qtA zr|z-yiTmcgB7>)rsEyF*lnCiPWT49j=|&F?Y$MKCTZcOwrah%Zy21)rz4w{p-#R zbOCJGk3D$@M|UnCx~fu)Z4FCotjJ+Y#K63+y0;=6$$n<)loPMT*yt#qb;4u3tw8-% z?rI6^;X*-gj1(-a%}2TJN|b8`>v$5PGKlb2L=2w+Vo|}@NC~2bd^2Y%`g`7!m)Uao z;0=DsM|Pbeme*la63rq_2H0(pi7oWF1DXpCoO?Y@$P!5(u0?{_El~I?nB)8Bqe2#7 zwV*#w4$ivC9)XVR{!^zR)=>dDAf3vC1}lVLb*gD@{y&+5^!fdh7%6r$2>aYT*3LNR zpDBIu^s|z1x$YAsOg}p(xn+wNgd#kcZ!;U}wcpw#Zm0E6vEl{c)?xaeOBLe0NkNE2zn4u0F=!>^Y_i`}mZl56F~=paL}_76)ByxRBL;jicnizVj}25Bq^a zz+gu4J45aj4|VWeW=P{wp6a>_GwXwnu{amdIffoofgU->sWk>WY~2bzT7}Wx6gZ^D zy2uY)g+K>9dB(HqYDJ{Ci@{^MbVr%8Q65VBFy|F*2 z{28lt7Pv8XyrpLHNF*#yeN~SEg2m;l;zLu3ZBcJU7rO1ZWZrc=l%@}-V?sp$8LxmD zW7kAx0^E*cLF@KX(b%ab1M#J;>F3vBLufB7J=9&jF8(?LJ{T2*4gjfp z^>&K%>nMD?Df>Tx|ByErSZT!fRUhp6V;*3LE=kCiMs|xEEx4(aAj_EsdM*VCxFT&t z1ap+@@ytivX$^4-r>FuZmQ_15OIB;}q&NnZ4iqOM55&kaqslMYCl(dfgh4^?iqm+6 zVixG=PHeB1pE>mB>Q7qgAqWaaE_jw*cH4-#XyCjgLKlfE6}B60k_4)bl-KHc=Z4}$ zjvhXYuRa`57-7ByFkeUmY>V-tO=Mgw^y{~J*~gPe8d~~twq5|FE~FqGBq52-FKByl zLha*`Ll^rA5D8pYR=eya>$Qf}anI3(Cbu+G zJ{Z$VBbJgqS^FT-CzGlo7ywQKcYm&Cjy38_Qm;Ki_`1C!TCIQOT$wd3 zU8|vV0_oWinmGB3O`3Fy-7 zn>EYn?DHn?UhFO*e@N$Jp!Uy98r^&b7eg+L-lda@ofkzL%A#mjMzmq03b4L1{2{zmvi3lac=o2EPvm-({eAD+B8r8FZ0?WB^@*DPYm9)uphZ@J$Bn zH!?W%ovqBpdx1@gv6g^_@=QQlaOTtM`bjNF217dX!Ul65aK7_~LvC|M?aPMePRUXWLrh1D%6roe6GLI@fg6_Ex)u^*gmH_U^Aod0=GfNQ~w_;)h6 zeZ=rT8T=nIh@$!bp$zVUQ-A{M-3NOBsu%%SXrz{UH)vv@3|iYE|5DX&zbf+rK+dXW z*6K12=DJp@H?v&{k>3HLHx4S0K>z^I^ci&H`Q-xz+D-U2!?+dE-Hdt$W+{rGOiw|E zfHeBcM+kKG&i5nEwEzI)zm+=sMbGdls8l!5Acy%&Rxq$WkmC#}^S6R;9JGHx{MCrN zC#(O2q6fJ;e>dWfbJp)xaQ94u zp8&qkZvlRuaBw&4-DrzHvy!rW%X&BV;$4Kh(K>!2JQw^|u{-W!+;y@4iBY9;m&d<4 z+}{Pf>zV!&a7gw40Q@-*x(j&Mq5CJ`mf9`g9e3`(x_AEpddKDWCn&z=E$CkyfbU}5 zb@ly;Ve$>*-xzl5*mxK8t~=yU&?Jz@F-X_H8}?U>-`ya8B7lLdgD`%sMgCcZ+%@8N z_rhNR26gWO{B$qs`F%s}SCo9KyC}EQ zu)Fp6ePioa06puw0JmoR73KGhpI=deZ0@4mP6O_m@%s+NuK>h$cL8qA_$$iqkGOwD zA#?a;%=q!Z`>qkcKY9E$`+(Cg*}p%Byqo*?Cp*99E_D4R_cw+9?H=^|y82fDOZQ(n z_}eP`t}(x_&VI$R_xwB7zaFE0pW^?D<>vi&te+G9yLSCPPy7|j$M^48KWB{+B2f2l Ra^Qjf8G~-}!TvXY{Xdisc>e$Z literal 0 HcmV?d00001