2024-05-31 09:40:38 +00:00
import { uniq , update } from "lodash" ;
2024-04-08 13:41:47 +00:00
import isEmpty from "lodash/isEmpty" ;
import omit from "lodash/omit" ;
2024-04-10 14:25:16 +00:00
import orderBy from "lodash/orderBy" ;
2024-04-08 13:41:47 +00:00
import set from "lodash/set" ;
import { action , computed , makeObservable , observable , runInAction } from "mobx" ;
import { computedFn } from "mobx-utils" ;
// types
import {
TInboxIssue ,
TInboxIssueCurrentTab ,
TInboxIssueFilter ,
TInboxIssueSorting ,
TInboxIssuePaginationInfo ,
TInboxIssueSortingOrderByQueryParam ,
} from "@plane/types" ;
2024-04-10 08:22:57 +00:00
// helpers
2024-04-10 14:25:16 +00:00
import { EInboxIssueCurrentTab , EInboxIssueStatus , EPastDurationFilters , getCustomDates } from "@/helpers/inbox.helper" ;
2024-04-08 13:41:47 +00:00
// services
import { InboxIssueService } from "@/services/inbox" ;
// root store
import { IInboxIssueStore , InboxIssueStore } from "@/store/inbox/inbox-issue.store" ;
import { RootStore } from "@/store/root.store" ;
2024-04-10 14:25:16 +00:00
type TLoader =
| "init-loading"
| "mutation-loading"
| "filter-loading"
| "pagination-loading"
| "issue-loading"
| undefined ;
2024-04-08 13:41:47 +00:00
export interface IProjectInboxStore {
currentTab : TInboxIssueCurrentTab ;
2024-04-15 07:19:14 +00:00
loader : TLoader ;
2024-04-08 13:41:47 +00:00
error : { message : string ; status : "init-error" | "pagination-error" } | undefined ;
2024-04-10 14:25:16 +00:00
currentInboxProjectId : string ;
2024-04-08 13:41:47 +00:00
inboxFilters : Partial < TInboxIssueFilter > ;
inboxSorting : Partial < TInboxIssueSorting > ;
inboxIssuePaginationInfo : TInboxIssuePaginationInfo | undefined ;
2024-04-10 10:38:31 +00:00
inboxIssues : Record < string , IInboxIssueStore > ; // issue_id -> IInboxIssueStore
2024-05-31 09:40:38 +00:00
inboxIssueIds : string [ ] ;
2024-04-08 13:41:47 +00:00
// computed
getAppliedFiltersCount : number ;
2024-05-31 09:40:38 +00:00
// computed functions
2024-04-15 07:19:14 +00:00
getIssueInboxByIssueId : ( issueId : string ) = > IInboxIssueStore ;
2024-05-31 09:40:38 +00:00
getIsIssueAvailable : ( inboxIssueId : string ) = > boolean ;
// helper actions
2024-04-10 14:25:16 +00:00
inboxIssueSorting : ( issues : IInboxIssueStore [ ] ) = > IInboxIssueStore [ ] ;
2024-04-08 13:41:47 +00:00
inboxIssueQueryParams : (
inboxFilters : Partial < TInboxIssueFilter > ,
inboxSorting : Partial < TInboxIssueSorting > ,
pagePerCount : number ,
paginationCursor : string
) = > Partial < Record < keyof TInboxIssueFilter , string > > ;
2024-05-31 09:40:38 +00:00
createOrUpdateInboxIssue : ( inboxIssues : TInboxIssue [ ] , workspaceSlug : string , projectId : string ) = > void ;
2024-04-08 13:41:47 +00:00
// actions
handleCurrentTab : ( tab : TInboxIssueCurrentTab ) = > void ;
handleInboxIssueFilters : < T extends keyof TInboxIssueFilter > ( key : T , value : TInboxIssueFilter [ T ] ) = > void ; // if user sends me undefined, I will remove the value from the filter key
handleInboxIssueSorting : < T extends keyof TInboxIssueSorting > ( key : T , value : TInboxIssueSorting [ T ] ) = > void ; // if user sends me undefined, I will remove the value from the filter key
fetchInboxIssues : ( workspaceSlug : string , projectId : string , loadingType? : TLoader ) = > Promise < void > ;
fetchInboxPaginationIssues : ( workspaceSlug : string , projectId : string ) = > Promise < void > ;
2024-04-11 15:58:59 +00:00
fetchInboxIssueById : ( workspaceSlug : string , projectId : string , inboxIssueId : string ) = > Promise < TInboxIssue > ;
2024-04-08 13:41:47 +00:00
createInboxIssue : (
workspaceSlug : string ,
projectId : string ,
data : Partial < TInboxIssue >
) = > Promise < TInboxIssue | undefined > ;
deleteInboxIssue : ( workspaceSlug : string , projectId : string , inboxIssueId : string ) = > Promise < void > ;
}
export class ProjectInboxStore implements IProjectInboxStore {
// constants
PER_PAGE_COUNT = 10 ;
// observables
2024-04-10 14:25:16 +00:00
currentTab : TInboxIssueCurrentTab = EInboxIssueCurrentTab . OPEN ;
2024-04-15 07:19:14 +00:00
loader : TLoader = "init-loading" ;
2024-04-08 13:41:47 +00:00
error : { message : string ; status : "init-error" | "pagination-error" } | undefined = undefined ;
2024-04-10 14:25:16 +00:00
currentInboxProjectId : string = "" ;
2024-04-08 13:41:47 +00:00
inboxFilters : Partial < TInboxIssueFilter > = {
2024-04-10 14:25:16 +00:00
status : [ EInboxIssueStatus . PENDING ] ,
2024-04-08 13:41:47 +00:00
} ;
inboxSorting : Partial < TInboxIssueSorting > = {
order_by : "issue__created_at" ,
sort_by : "desc" ,
} ;
inboxIssuePaginationInfo : TInboxIssuePaginationInfo | undefined = undefined ;
inboxIssues : Record < string , IInboxIssueStore > = { } ;
2024-05-31 09:40:38 +00:00
inboxIssueIds : string [ ] = [ ] ;
2024-04-08 13:41:47 +00:00
// services
inboxIssueService ;
constructor ( private store : RootStore ) {
makeObservable ( this , {
currentTab : observable.ref ,
2024-04-15 07:19:14 +00:00
loader : observable.ref ,
2024-04-10 14:25:16 +00:00
error : observable ,
currentInboxProjectId : observable.ref ,
2024-04-08 13:41:47 +00:00
inboxFilters : observable ,
inboxSorting : observable ,
inboxIssuePaginationInfo : observable ,
inboxIssues : observable ,
2024-05-31 09:40:38 +00:00
inboxIssueIds : observable ,
2024-04-08 13:41:47 +00:00
// computed
getAppliedFiltersCount : computed ,
// actions
handleInboxIssueFilters : action ,
handleInboxIssueSorting : action ,
fetchInboxIssues : action ,
fetchInboxPaginationIssues : action ,
fetchInboxIssueById : action ,
createInboxIssue : action ,
deleteInboxIssue : action ,
} ) ;
this . inboxIssueService = new InboxIssueService ( ) ;
}
// computed
get getAppliedFiltersCount() {
let count = 0 ;
this . inboxFilters != undefined &&
Object . keys ( this . inboxFilters ) . forEach ( ( key ) = > {
const filterKey = key as keyof TInboxIssueFilter ;
if ( this . inboxFilters [ filterKey ] && this . inboxFilters ? . [ filterKey ] )
count = count + ( this . inboxFilters ? . [ filterKey ] ? . length ? ? 0 ) ;
} ) ;
return count ;
}
2024-04-15 07:19:14 +00:00
getIssueInboxByIssueId = computedFn ( ( issueId : string ) = > this . inboxIssues ? . [ issueId ] ) ;
2024-04-08 13:41:47 +00:00
2024-05-31 09:40:38 +00:00
getIsIssueAvailable = computedFn ( ( inboxIssueId : string ) = > {
if ( ! this . inboxIssueIds ) return true ;
return this . inboxIssueIds . includes ( inboxIssueId ) ;
} ) ;
2024-04-10 14:25:16 +00:00
// helpers
inboxIssueSorting = ( issues : IInboxIssueStore [ ] ) = > {
2024-04-10 15:59:21 +00:00
let inboxIssues : IInboxIssueStore [ ] = issues ;
inboxIssues = orderBy ( inboxIssues , "issue.sequence_id" , "desc" ) ;
2024-04-10 14:25:16 +00:00
if ( this . inboxSorting ? . order_by && this . inboxSorting ? . sort_by ) {
switch ( this . inboxSorting . order_by ) {
case "issue__created_at" :
2024-04-10 15:59:21 +00:00
if ( this . inboxSorting . sort_by === "desc" ) inboxIssues = orderBy ( inboxIssues , "issue.created_at" , "desc" ) ;
else inboxIssues = orderBy ( inboxIssues , "issue.created_at" , "asc" ) ;
2024-04-10 14:25:16 +00:00
case "issue__updated_at" :
2024-04-10 15:59:21 +00:00
if ( this . inboxSorting . sort_by === "desc" ) inboxIssues = orderBy ( inboxIssues , "issue.updated_at" , "desc" ) ;
else inboxIssues = orderBy ( inboxIssues , "issue.updated_at" , "asc" ) ;
2024-04-10 14:25:16 +00:00
case "issue__sequence_id" :
2024-04-10 15:59:21 +00:00
if ( this . inboxSorting . sort_by === "desc" ) inboxIssues = orderBy ( inboxIssues , "issue.sequence_id" , "desc" ) ;
else inboxIssues = orderBy ( inboxIssues , "issue.sequence_id" , "asc" ) ;
2024-04-10 14:25:16 +00:00
default :
2024-04-10 15:59:21 +00:00
inboxIssues = inboxIssues ;
2024-04-10 14:25:16 +00:00
}
}
return inboxIssues ;
} ;
2024-04-08 13:41:47 +00:00
inboxIssueQueryParams = (
inboxFilters : Partial < TInboxIssueFilter > ,
inboxSorting : Partial < TInboxIssueSorting > ,
pagePerCount : number ,
paginationCursor : string
) = > {
const filters : Partial < Record < keyof TInboxIssueFilter , string > > = { } ;
! isEmpty ( inboxFilters ) &&
Object . keys ( inboxFilters ) . forEach ( ( key ) = > {
const filterKey = key as keyof TInboxIssueFilter ;
2024-04-10 08:22:57 +00:00
if ( inboxFilters [ filterKey ] && inboxFilters [ filterKey ] ? . length ) {
if ( [ "created_at" , "updated_at" ] . includes ( filterKey ) && ( inboxFilters [ filterKey ] || [ ] ) ? . length > 0 ) {
const appliedDateFilters : string [ ] = [ ] ;
inboxFilters [ filterKey ] ? . forEach ( ( value ) = > {
const dateValue = value as EPastDurationFilters ;
appliedDateFilters . push ( getCustomDates ( dateValue ) ) ;
} ) ;
filters [ filterKey ] = appliedDateFilters ? . join ( "," ) ;
} else filters [ filterKey ] = inboxFilters [ filterKey ] ? . join ( "," ) ;
}
2024-04-08 13:41:47 +00:00
} ) ;
const sorting : TInboxIssueSortingOrderByQueryParam = {
order_by : "-issue__created_at" ,
} ;
if ( inboxSorting ? . order_by && inboxSorting ? . sort_by ) {
switch ( inboxSorting . order_by ) {
case "issue__created_at" :
if ( inboxSorting . sort_by === "desc" ) sorting . order_by = ` -issue__created_at ` ;
else sorting . order_by = "issue__created_at" ;
break ;
case "issue__updated_at" :
if ( inboxSorting . sort_by === "desc" ) sorting . order_by = ` -issue__updated_at ` ;
else sorting . order_by = "issue__updated_at" ;
break ;
case "issue__sequence_id" :
if ( inboxSorting . sort_by === "desc" ) sorting . order_by = ` -issue__sequence_id ` ;
else sorting . order_by = "issue__sequence_id" ;
break ;
default :
sorting . order_by = "-issue__created_at" ;
break ;
}
}
return {
. . . filters ,
. . . sorting ,
per_page : pagePerCount ,
cursor : paginationCursor ,
} ;
} ;
2024-05-31 09:40:38 +00:00
createOrUpdateInboxIssue = ( inboxIssues : TInboxIssue [ ] , workspaceSlug : string , projectId : string ) = > {
if ( inboxIssues && inboxIssues . length > 0 ) {
inboxIssues . forEach ( ( inbox : TInboxIssue ) = > {
const inboxIssueDetail = this . getIssueInboxByIssueId ( inbox ? . issue ? . id ) ;
if ( inboxIssueDetail )
update ( this . inboxIssues , [ inbox ? . issue ? . id ] , ( existingInboxIssue ) = > ( {
. . . existingInboxIssue ,
. . . inbox ,
issue : {
. . . existingInboxIssue ? . issue ,
. . . inbox ? . issue ,
} ,
} ) ) ;
else
set ( this . inboxIssues , [ inbox ? . issue ? . id ] , new InboxIssueStore ( workspaceSlug , projectId , inbox , this . store ) ) ;
} ) ;
}
} ;
2024-04-08 13:41:47 +00:00
// actions
handleCurrentTab = ( tab : TInboxIssueCurrentTab ) = > {
2024-05-31 09:40:38 +00:00
runInAction ( ( ) = > {
set ( this , "currentTab" , tab ) ;
set ( this , "inboxFilters" , undefined ) ;
set ( this , [ "inboxSorting" , "order_by" ] , "issue__created_at" ) ;
set ( this , [ "inboxSorting" , "sort_by" ] , "desc" ) ;
set ( this , [ "inboxIssueIds" ] , [ ] ) ;
set ( this , [ "inboxIssuePaginationInfo" ] , undefined ) ;
if ( tab === "closed" ) set ( this , [ "inboxFilters" , "status" ] , [ - 1 , 1 , 2 ] ) ;
else set ( this , [ "inboxFilters" , "status" ] , [ - 2 ] ) ;
} ) ;
2024-05-08 17:31:20 +00:00
const { workspaceSlug , projectId } = this . store . router ;
2024-04-08 13:41:47 +00:00
if ( workspaceSlug && projectId ) this . fetchInboxIssues ( workspaceSlug , projectId , "filter-loading" ) ;
} ;
handleInboxIssueFilters = < T extends keyof TInboxIssueFilter > ( key : T , value : TInboxIssueFilter [ T ] ) = > {
2024-05-31 09:40:38 +00:00
runInAction ( ( ) = > {
set ( this . inboxFilters , key , value ) ;
set ( this , [ "inboxIssuePaginationInfo" ] , undefined ) ;
} ) ;
2024-05-08 17:31:20 +00:00
const { workspaceSlug , projectId } = this . store . router ;
2024-04-08 13:41:47 +00:00
if ( workspaceSlug && projectId ) this . fetchInboxIssues ( workspaceSlug , projectId , "filter-loading" ) ;
} ;
handleInboxIssueSorting = < T extends keyof TInboxIssueSorting > ( key : T , value : TInboxIssueSorting [ T ] ) = > {
2024-05-31 09:40:38 +00:00
runInAction ( ( ) = > {
set ( this . inboxSorting , key , value ) ;
set ( this , [ "inboxIssuePaginationInfo" ] , undefined ) ;
} ) ;
2024-05-08 17:31:20 +00:00
const { workspaceSlug , projectId } = this . store . router ;
2024-04-08 13:41:47 +00:00
if ( workspaceSlug && projectId ) this . fetchInboxIssues ( workspaceSlug , projectId , "filter-loading" ) ;
} ;
/ * *
* @description fetch inbox issues with paginated data
* @param workspaceSlug
* @param projectId
* /
fetchInboxIssues = async ( workspaceSlug : string , projectId : string , loadingType : TLoader = undefined ) = > {
try {
2024-04-10 14:25:16 +00:00
if ( this . currentInboxProjectId != projectId ) {
2024-05-31 09:40:38 +00:00
runInAction ( ( ) = > {
set ( this , [ "currentInboxProjectId" ] , projectId ) ;
set ( this , [ "inboxIssues" ] , { } ) ;
set ( this , [ "inboxIssueIds" ] , [ ] ) ;
set ( this , [ "inboxIssuePaginationInfo" ] , undefined ) ;
} ) ;
2024-04-10 14:25:16 +00:00
}
2024-05-31 09:40:38 +00:00
if ( Object . keys ( this . inboxIssueIds ) . length === 0 ) this . loader = "init-loading" ;
2024-04-15 07:19:14 +00:00
else this . loader = "mutation-loading" ;
if ( loadingType ) this . loader = loadingType ;
2024-04-08 13:41:47 +00:00
const queryParams = this . inboxIssueQueryParams (
this . inboxFilters ,
this . inboxSorting ,
this . PER_PAGE_COUNT ,
` ${ this . PER_PAGE_COUNT } :0:0 `
) ;
const { results , . . . paginationInfo } = await this . inboxIssueService . list ( workspaceSlug , projectId , queryParams ) ;
runInAction ( ( ) = > {
2024-04-15 07:19:14 +00:00
this . loader = undefined ;
2024-04-08 13:41:47 +00:00
set ( this , "inboxIssuePaginationInfo" , paginationInfo ) ;
2024-05-31 09:40:38 +00:00
if ( results ) {
const issueIds = results . map ( ( value ) = > value ? . issue ? . id ) ;
set ( this , [ "inboxIssueIds" ] , issueIds ) ;
this . createOrUpdateInboxIssue ( results , workspaceSlug , projectId ) ;
}
2024-04-08 13:41:47 +00:00
} ) ;
} catch ( error ) {
console . error ( "Error fetching the inbox issues" , error ) ;
2024-04-15 07:19:14 +00:00
this . loader = undefined ;
2024-04-08 13:41:47 +00:00
this . error = {
message : "Error fetching the inbox issues please try again later." ,
status : "init-error" ,
} ;
throw error ;
}
} ;
/ * *
* @description fetch inbox issues with paginated data
* @param workspaceSlug
* @param projectId
* /
fetchInboxPaginationIssues = async ( workspaceSlug : string , projectId : string ) = > {
try {
if (
2024-04-10 14:25:16 +00:00
this . inboxIssuePaginationInfo &&
( ! this . inboxIssuePaginationInfo ? . total_results ||
( this . inboxIssuePaginationInfo ? . total_results &&
2024-05-31 09:40:38 +00:00
this . inboxIssueIds . length < this . inboxIssuePaginationInfo ? . total_results ) )
2024-04-08 13:41:47 +00:00
) {
2024-04-15 07:19:14 +00:00
this . loader = "pagination-loading" ;
2024-04-08 13:41:47 +00:00
const queryParams = this . inboxIssueQueryParams (
this . inboxFilters ,
this . inboxSorting ,
this . PER_PAGE_COUNT ,
this . inboxIssuePaginationInfo ? . next_cursor || ` ${ this . PER_PAGE_COUNT } :0:0 `
) ;
const { results , . . . paginationInfo } = await this . inboxIssueService . list ( workspaceSlug , projectId , queryParams ) ;
runInAction ( ( ) = > {
2024-04-15 07:19:14 +00:00
this . loader = undefined ;
2024-04-08 13:41:47 +00:00
set ( this , "inboxIssuePaginationInfo" , paginationInfo ) ;
2024-05-31 09:40:38 +00:00
if ( results && results . length > 0 ) {
const issueIds = results . map ( ( value ) = > value ? . issue ? . id ) ;
update ( this , [ "inboxIssueIds" ] , ( ids ) = > uniq ( [ . . . ids , . . . issueIds ] ) ) ;
this . createOrUpdateInboxIssue ( results , workspaceSlug , projectId ) ;
}
2024-04-08 13:41:47 +00:00
} ) ;
} else set ( this , [ "inboxIssuePaginationInfo" , "next_page_results" ] , false ) ;
} catch ( error ) {
console . error ( "Error fetching the inbox issues" , error ) ;
2024-04-15 07:19:14 +00:00
this . loader = undefined ;
2024-04-08 13:41:47 +00:00
this . error = {
message : "Error fetching the paginated inbox issues please try again later." ,
status : "pagination-error" ,
} ;
throw error ;
}
} ;
/ * *
* @description fetch inbox issue with issue id
* @param workspaceSlug
* @param projectId
* @param inboxIssueId
* /
2024-04-11 15:58:59 +00:00
fetchInboxIssueById = async (
workspaceSlug : string ,
projectId : string ,
inboxIssueId : string
) : Promise < TInboxIssue > = > {
2024-04-08 13:41:47 +00:00
try {
2024-04-15 07:19:14 +00:00
this . loader = "issue-loading" ;
2024-04-08 13:41:47 +00:00
const inboxIssue = await this . inboxIssueService . retrieve ( workspaceSlug , projectId , inboxIssueId ) ;
const issueId = inboxIssue ? . issue ? . id || undefined ;
if ( inboxIssue && issueId ) {
2024-04-10 14:25:16 +00:00
runInAction ( ( ) = > {
set ( this . inboxIssues , [ issueId ] , new InboxIssueStore ( workspaceSlug , projectId , inboxIssue , this . store ) ) ;
2024-04-15 14:15:47 +00:00
set ( this , "loader" , undefined ) ;
2024-04-10 14:25:16 +00:00
} ) ;
2024-05-31 09:40:38 +00:00
await Promise . all ( [
// fetching reactions
this . store . issue . issueDetail . fetchReactions ( workspaceSlug , projectId , issueId ) ,
// fetching activity
this . store . issue . issueDetail . fetchActivities ( workspaceSlug , projectId , issueId ) ,
// fetching comments
this . store . issue . issueDetail . fetchComments ( workspaceSlug , projectId , issueId ) ,
// fetching attachments
this . store . issue . issueDetail . fetchAttachments ( workspaceSlug , projectId , issueId ) ,
] ) ;
2024-04-08 13:41:47 +00:00
}
2024-04-11 15:58:59 +00:00
return inboxIssue ;
} catch ( error ) {
2024-04-08 13:41:47 +00:00
console . error ( "Error fetching the inbox issue with inbox issue id" ) ;
2024-04-15 07:19:14 +00:00
this . loader = undefined ;
2024-04-11 15:58:59 +00:00
throw error ;
2024-04-08 13:41:47 +00:00
}
} ;
/ * *
* @description create inbox issue
* @param workspaceSlug
* @param projectId
* @param data
* /
createInboxIssue = async ( workspaceSlug : string , projectId : string , data : Partial < TInboxIssue > ) = > {
try {
const inboxIssueResponse = await this . inboxIssueService . create ( workspaceSlug , projectId , data ) ;
if ( inboxIssueResponse )
runInAction ( ( ) = > {
2024-05-31 09:40:38 +00:00
update ( this , [ "inboxIssueIds" ] , ( ids ) = > [ . . . ids , inboxIssueResponse ? . issue ? . id ] ) ;
2024-04-08 13:41:47 +00:00
set (
this . inboxIssues ,
2024-04-10 10:38:31 +00:00
[ inboxIssueResponse ? . issue ? . id ] ,
new InboxIssueStore ( workspaceSlug , projectId , inboxIssueResponse , this . store )
2024-04-08 13:41:47 +00:00
) ;
set (
this ,
[ "inboxIssuePaginationInfo" , "total_results" ] ,
( this . inboxIssuePaginationInfo ? . total_results || 0 ) + 1
) ;
} ) ;
return inboxIssueResponse ;
} catch {
console . error ( "Error creating the inbox issue" ) ;
}
} ;
/ * *
* @description delete inbox issue
* @param workspaceSlug
* @param projectId
* @param inboxIssueId
* /
deleteInboxIssue = async ( workspaceSlug : string , projectId : string , inboxIssueId : string ) = > {
const currentIssue = this . inboxIssues ? . [ inboxIssueId ] ;
try {
if ( ! currentIssue ) return ;
runInAction ( ( ) = > {
set (
this ,
[ "inboxIssuePaginationInfo" , "total_results" ] ,
( this . inboxIssuePaginationInfo ? . total_results || 0 ) - 1
) ;
set ( this , "inboxIssues" , omit ( this . inboxIssues , inboxIssueId ) ) ;
2024-05-31 09:40:38 +00:00
set (
this ,
[ "inboxIssueIds" ] ,
this . inboxIssueIds . filter ( ( id ) = > id !== inboxIssueId )
) ;
2024-04-08 13:41:47 +00:00
} ) ;
await this . inboxIssueService . destroy ( workspaceSlug , projectId , inboxIssueId ) ;
} catch {
console . error ( "Error removing the inbox issue" ) ;
set ( this . inboxIssues , [ inboxIssueId ] , currentIssue ) ;
2024-05-31 09:40:38 +00:00
set ( this , [ "inboxIssuePaginationInfo" , "total_results" ] , ( this . inboxIssuePaginationInfo ? . total_results || 0 ) + 1 ) ;
set ( this , [ "inboxIssueIds" ] , [ . . . this . inboxIssueIds , inboxIssueId ] ) ;
2024-04-08 13:41:47 +00:00
}
} ;
}