// (C) Copyright 2015 Moodle Pty Ltd. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. import { ContextLevel, CoreCacheUpdateFrequency } from '@/core/constants'; import { Injectable } from '@angular/core'; import { CoreSiteWSPreSets } from '@classes/sites/authenticated-site'; import { CoreFileUploaderStoreFilesResult } from '@features/fileuploader/services/fileuploader'; import { CoreTagItem } from '@features/tag/services/tag'; import { CoreUser, CoreUserProfile } from '@features/user/services/user'; import { CoreFileEntry, CoreFileHelper } from '@services/file-helper'; import { CoreNetwork } from '@services/network'; import { CoreSites, CoreSitesCommonWSOptions } from '@services/sites'; import { CoreTimeUtils } from '@services/utils/time'; import { CoreUtils } from '@services/utils/utils'; import { CoreStatusWithWarningsWSResponse, CoreWSExternalFile, CoreWSExternalWarning } from '@services/ws'; import { makeSingleton } from '@singletons'; import { AddonBlogOffline, AddonBlogOfflineEntry } from './blog-offline'; import { CorePromiseUtils } from '@singletons/promise-utils'; const ROOT_CACHE_KEY = 'addonBlog:'; /** * Service to handle blog entries. */ @Injectable({ providedIn: 'root' }) export class AddonBlogProvider { static readonly ENTRIES_PER_PAGE = 10; static readonly COMPONENT = 'blog'; /** * Returns whether or not the blog plugin is enabled for a certain site. * * This method is called quite often and thus should only perform a quick * check, we should not be calling WS from here. * * @param siteId Site ID. If not defined, current site. * @returns Promise resolved with true if enabled, resolved with false or rejected otherwise. * @since 3.6 */ async isPluginEnabled(siteId?: string): Promise { const site = await CoreSites.getSite(siteId); return site.wsAvailable('core_blog_get_entries') && site.canUseAdvancedFeature('enableblogs'); } /** * Get the cache key for the blog entries. * * @param filter Filter to apply on search. * @returns Cache key. */ getEntriesCacheKey(filter: AddonBlogFilter = {}): string { return ROOT_CACHE_KEY + CoreUtils.sortAndStringify(filter); } /** * Get blog entries. * * @param filter Filter to apply on search. * @param options WS Options. * @returns Promise to be resolved when the entries are retrieved. */ async getEntries(filter: AddonBlogFilter = {}, options?: AddonBlogGetEntriesOptions): Promise { const site = await CoreSites.getSite(options?.siteId); const data: CoreBlogGetEntriesWSParams = { filters: CoreUtils.objectToArrayOfObjects(filter, 'name', 'value'), page: options?.page ?? 0, perpage: AddonBlogProvider.ENTRIES_PER_PAGE, }; const preSets: CoreSiteWSPreSets = { cacheKey: this.getEntriesCacheKey(filter), updateFrequency: CoreCacheUpdateFrequency.SOMETIMES, ...CoreSites.getReadingStrategyPreSets(options?.readingStrategy), }; return site.read('core_blog_get_entries', data, preSets); } /** * Create a new entry. * * @param params WS Params. * @param siteId Site ID where the entry should be created. * @returns Entry id. * @since 4.4 */ async addEntry( { created, forceOffline, ...params }: AddonBlogAddEntryWSParams & { created: number; forceOffline?: boolean }, siteId?: string, ): Promise { const site = await CoreSites.getSite(siteId); const storeOffline = async (): Promise => { await AddonBlogOffline.addOfflineEntry({ ...params, userid: site.getUserId(), lastmodified: created, options: JSON.stringify(params.options), created, }); }; if (forceOffline || !CoreNetwork.isOnline()) { return await storeOffline(); } try { await this.addEntryOnline(params, siteId); } catch (error) { if (!CoreUtils.isWebServiceError(error)) { // Couldn't connect to server, store in offline. return await storeOffline(); } // The WebService has thrown an error, reject. throw error; } } /** * Add entry online. * * @param wsParams Params expected by the webservice. * @param siteId Site ID. */ async addEntryOnline(wsParams: AddonBlogAddEntryWSParams, siteId?: string): Promise { const site = await CoreSites.getSite(siteId); await site.write('core_blog_add_entry', wsParams); } /** * Update an entry. * * @param params WS Params. * @param siteId Site ID of the entry. * @since 4.4 * @returns void */ async updateEntry( { forceOffline, created, ...params }: AddonBlogUpdateEntryParams, siteId?: string, ): Promise { const site = await CoreSites.getSite(siteId); const storeOffline = async (): Promise => { const content = { subject: params.subject, summary: params.summary, summaryformat: params.summaryformat, userid: site.getUserId(), lastmodified: CoreTimeUtils.timestamp(), options: JSON.stringify(params.options), created, }; await AddonBlogOffline.addOfflineEntry({ ...content, id: params.entryid }); }; if (forceOffline || !CoreNetwork.isOnline()) { return await storeOffline(); } try { await this.updateEntryOnline(params, siteId); } catch (error) { if (!CoreUtils.isWebServiceError(error)) { // Couldn't connect to server, store in offline. return await storeOffline(); } // The WebService has thrown an error, reject. throw error; } } /** * Update entry online. * * @param wsParams Params expected by the webservice. * @param siteId Site ID. */ async updateEntryOnline(wsParams: AddonBlogUpdateEntryWSParams, siteId?: string): Promise { const site = await CoreSites.getSite(siteId); await site.write('core_blog_update_entry', wsParams); } /** * Prepare entry for edition by entry id. * * @param params WS Params. * @param siteId Site ID of the entry. * @returns WS Response * @since 4.4 */ async prepareEntryForEdition( params: AddonBlogPrepareEntryForEditionWSParams, siteId?: string, ): Promise { const site = await CoreSites.getSite(siteId); return await site.write('core_blog_prepare_entry_for_edition', params); } /** * Delete entry by id. * * @param params WS params. * @param siteId Site ID of the entry. * @returns Entry deleted successfully or not. * @since 4.4 */ async deleteEntry({ subject, ...params }: AddonBlogDeleteEntryWSParams & { subject: string }, siteId?: string): Promise { try { if (!CoreNetwork.isOnline()) { return await AddonBlogOffline.markEntryAsRemoved({ id: params.entryid, subject }, siteId); } await this.deleteEntryOnline(params, siteId); await CorePromiseUtils.ignoreErrors(AddonBlogOffline.unmarkEntryAsRemoved(params.entryid)); } catch (error) { if (!CoreUtils.isWebServiceError(error)) { // Couldn't connect to server, store in offline. return await AddonBlogOffline.markEntryAsRemoved({ id: params.entryid, subject }, siteId); } throw error; } } /** * Delete entry online. * * @param wsParams Params expected by the webservice. * @param siteId Site ID. */ async deleteEntryOnline(wsParams: AddonBlogDeleteEntryWSParams, siteId?: string): Promise { const site = await CoreSites.getSite(siteId); await site.write('core_blog_delete_entry', wsParams); } /** * Invalidate blog entries WS call. * * @param filter Filter to apply on search * @param siteId Site ID. If not defined, current site. * @returns Promise resolved when data is invalidated. */ async invalidateEntries(filter: AddonBlogFilter = {}, siteId?: string): Promise { const site = await CoreSites.getSite(siteId); await site.invalidateWsCacheForKey(this.getEntriesCacheKey(filter)); } /** * Is editing blog entry enabled. * * @param siteId Site ID. * @returns is enabled or not. * @since 4.4 */ async isEditingEnabled(siteId?: string): Promise { const site = await CoreSites.getSite(siteId); return site.wsAvailable('core_blog_update_entry'); } /** * Trigger the blog_entries_viewed event. * * @param filter Filter to apply on search. * @param siteId Site ID. If not defined, current site. * @returns Promise to be resolved when done. */ async logView(filter: AddonBlogFilter = {}, siteId?: string): Promise { const site = await CoreSites.getSite(siteId); const data: AddonBlogViewEntriesWSParams = { filters: CoreUtils.objectToArrayOfObjects(filter, 'name', 'value'), }; return site.write('core_blog_view_entries', data); } /** * Format local stored entries to required data structure. * * @param offlineEntry Offline entry data. * @param entry Entry. * @returns Formatted entry. */ async formatOfflineEntry( offlineEntry: AddonBlogOfflineEntry, entry?: AddonBlogPostFormatted, ): Promise { const options: AddonBlogAddEntryOption[] = JSON.parse(offlineEntry.options); const moduleId = options?.find(option => option.name === 'modassoc')?.value as number | undefined; const courseId = options?.find(option => option.name === 'courseassoc')?.value as number | undefined; const tags = options?.find(option => option.name === 'tags')?.value as string | undefined; const publishState = options?.find(option => option.name === 'publishstate')?.value as AddonBlogPublishState ?? AddonBlogPublishState.draft; const user = await CorePromiseUtils.ignoreErrors(CoreUser.getProfile(offlineEntry.userid, courseId, true)); const folder = 'id' in offlineEntry && offlineEntry.id ? { id: offlineEntry.id } : { created: offlineEntry.created }; const offlineFiles = await AddonBlogOffline.getOfflineFiles(folder); const optionsFiles = this.getAttachmentFilesFromOptions(options); const attachmentFiles = [...optionsFiles.online, ...offlineFiles]; return { ...offlineEntry, publishstate: publishState, publishTranslated: this.getPublishTranslated(publishState), user, tags: tags?.length ? JSON.parse(tags) : [], coursemoduleid: moduleId ?? 0, courseid: courseId ?? 0, attachmentfiles: attachmentFiles, userid: user?.id ?? 0, moduleid: moduleId ?? 0, summaryfiles: [], uniquehash: '', module: entry?.module, groupid: 0, content: offlineEntry.summary, updatedOffline: true, }; } /** * Retrieves publish state translated. * * @param state Publish state. * @returns Translated state. */ getPublishTranslated(state?: string): string { switch (state) { case 'draft': return 'publishtonoone'; case 'site': return 'publishtosite'; case 'public': return 'publishtoworld'; default: return 'privacy:unknown'; } } /** * Format provided entry to AddonBlogPostFormatted. */ async formatEntry(entry: AddonBlogPostFormatted): Promise { entry.publishTranslated = this.getPublishTranslated(entry.publishstate); // Calculate the context. This code was inspired by calendar events, Moodle doesn't do this for blogs. if (entry.moduleid || entry.coursemoduleid) { entry.contextLevel = ContextLevel.MODULE; entry.contextInstanceId = entry.moduleid || entry.coursemoduleid; } else if (entry.courseid) { entry.contextLevel = ContextLevel.COURSE; entry.contextInstanceId = entry.courseid; } else { entry.contextLevel = ContextLevel.USER; entry.contextInstanceId = entry.userid; } entry.summary = CoreFileHelper.replacePluginfileUrls(entry.summary, entry.summaryfiles || []); entry.user = await CorePromiseUtils.ignoreErrors(CoreUser.getProfile(entry.userid, entry.courseid, true)); } /** * Get attachments files from options object. * * @param options Entry options. * @returns attachmentsId. */ getAttachmentFilesFromOptions(options: AddonBlogAddEntryOption[]): CoreFileUploaderStoreFilesResult { const attachmentsId = options.find(option => option.name === 'attachmentsid'); if (!attachmentsId) { return { online: [], offline: 0 }; } switch(typeof attachmentsId.value) { case 'object': return attachmentsId.value; case 'string': return JSON.parse(attachmentsId.value); default: return { online: [], offline: 0 }; } } } export const AddonBlog = makeSingleton(AddonBlogProvider); /** * Params of core_blog_get_entries WS. */ type CoreBlogGetEntriesWSParams = { filters?: { // Parameters to filter blog listings. name: string; // The expected keys (value format) are: // tag PARAM_NOTAGS blog tag // tagid PARAM_INT blog tag id // userid PARAM_INT blog author (userid) // cmid PARAM_INT course module id // entryid PARAM_INT entry id // groupid PARAM_INT group id // courseid PARAM_INT course id // search PARAM_RAW search term. value: string; // The value of the filter. }[]; page?: number; // The blog page to return. perpage?: number; // The number of posts to return per page. }; /** * Data returned by core_blog_get_entries WS. */ export type CoreBlogGetEntriesWSResponse = { entries: AddonBlogPost[]; totalentries: number; // The total number of entries found. warnings?: CoreWSExternalWarning[]; }; /** * Data returned by blog's post_exporter. */ export interface AddonBlogPost { id: number; // Post/entry id. module: string; // Where it was published the post (blog, blog_external...). userid: number; // Post author. courseid: number; // Course where the post was created. groupid: number; // Group post was created for. moduleid: number; // Module id where the post was created (not used anymore). coursemoduleid: number; // Course module id where the post was created. subject: string; // Post subject. summary: string; // Post summary. summaryformat?: number; // Summary format (1 = HTML, 0 = MOODLE, 2 = PLAIN or 4 = MARKDOWN). content: string; // Post content. uniquehash: string; // Post unique hash. rating: number; // Post rating. format: number; // Post content format. attachment: string; // Post atachment. publishstate: AddonBlogPublishState; // Post publish state. lastmodified: number; // When it was last modified. created: number; // When it was created. usermodified: number; // User that updated the post. summaryfiles: CoreWSExternalFile[]; // Summaryfiles. attachmentfiles?: CoreWSExternalFile[]; // Attachmentfiles. tags?: CoreTagItem[]; // @since 3.7. Tags. } /** * Params of core_blog_view_entries WS. */ type AddonBlogViewEntriesWSParams = { filters?: { // Parameters used in the filter of view_entries. name: string; // The expected keys (value format) are: // tag PARAM_NOTAGS blog tag // tagid PARAM_INT blog tag id // userid PARAM_INT blog author (userid) // cmid PARAM_INT course module id // entryid PARAM_INT entry id // groupid PARAM_INT group id // courseid PARAM_INT course id // search PARAM_RAW search term. value: string; // The value of the filter. }[]; }; export type AddonBlogFilter = { tag?: string; // Blog tag tagid?: number; // Blog tag id userid?: number; // Blog author (userid) cmid?: number; // Course module id entryid?: number; // Entry id groupid?: number; // Group id courseid?: number; // Course id search?: string; // Search term. }; /** * core_blog_add_entry & core_blog_update_entry ws params. */ export type AddonBlogAddEntryWSParams = { subject: string; summary: string; summaryformat: number; options: AddonBlogAddEntryOption[]; }; export type AddonBlogUpdateEntryWSParams = AddonBlogAddEntryWSParams & ({ entryid: number }); /** * Add entry options. */ export type AddonBlogAddEntryOption = { name: 'inlineattachmentsid' | 'attachmentsid' | 'publishstate' | 'courseassoc' | 'modassoc' | 'tags'; value: string | number | CoreFileUploaderStoreFilesResult; }; /** * core_blog_prepare_entry_for_edition ws params. */ export type AddonBlogPrepareEntryForEditionWSResponse = { inlineattachmentsid: number; attachmentsid: number; areas: AddonBlogPrepareEntryForEditionArea[]; warnings: string[]; }; export type AddonBlogPrepareEntryForEditionWSParams = { entryid: number; }; /** * core_blog_prepare_entry_for_edition Area object. */ export type AddonBlogPrepareEntryForEditionArea = { area: string; options: AddonBlogPrepareEntryForEditionOption[]; }; /** * core_blog_prepare_entry_for_edition Option object. */ export type AddonBlogPrepareEntryForEditionOption = { name: string; value: unknown; }; export type AddonBlogDeleteEntryWSParams = { entryid: number; }; export type AddonBlogDeleteEntryWSResponse = { status: boolean; // Status: true only if we set the policyagreed to 1 for the user. warnings?: CoreWSExternalWarning[]; }; export type AddonBlogGetEntriesOptions = CoreSitesCommonWSOptions & { page?: number; }; export type AddonBlogUndoDelete = { created: number } | { id: number }; export const AddonBlogPublishState = { draft: 'draft', site: 'site', public: 'public' } as const; // eslint-disable-next-line @typescript-eslint/no-redeclare export type AddonBlogPublishState = typeof AddonBlogPublishState[keyof typeof AddonBlogPublishState]; /** * Blog post with some calculated data. */ export type AddonBlogPostFormatted = Omit< AddonBlogPost, 'attachment' | 'attachmentfiles' | 'usermodified' | 'format' | 'rating' | 'module' > & { publishTranslated?: string; // Calculated in the app. Key of the string to translate the publish state of the post. user?: CoreUserProfile; // Calculated in the app. Data of the user that wrote the post. contextLevel?: ContextLevel; // Calculated in the app. The context level of the entry. contextInstanceId?: number; // Calculated in the app. The context instance id. coursemoduleid: number; // Course module id where the post was created. attachmentfiles?: CoreFileEntry[]; // Attachmentfiles. module?: string; deleted?: boolean; updatedOffline?: boolean; }; export type AddonBlogOfflinePostFormatted = Omit; export type AddonBlogUpdateEntryParams = AddonBlogUpdateEntryWSParams & { attachments?: string; forceOffline?: boolean; created: number; };