Skip to content
303 changes: 303 additions & 0 deletions src/CloudCode.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,306 @@
import type ParseObject from './ParseObject';
import type ParseUser from './ParseUser';
import type ParseFile from './ParseFile';
import type ParseQuery from './ParseQuery';

// ============================================================================
// Request Interfaces
// ============================================================================
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would remove these type / style of comments.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done


export interface FunctionRequest<T = Record<string, any>> {
installationId?: string;
master: boolean;
user?: ParseUser;
params: T;
ip: string;
headers: Record<string, string>;
log: any;
functionName: string;
context: Record<string, unknown>;
config: any;
}

export interface FunctionResponse {
success: (result?: any) => void;
error: (message?: string) => void;
status: (code: number) => FunctionResponse;
header: (name: string, value: string) => FunctionResponse;
}

export interface TriggerRequest<T extends ParseObject = ParseObject> {
installationId?: string;
master: boolean;
isChallenge: boolean;
user?: ParseUser;
object: T;
original?: T;
ip: string;
headers: Record<string, string>;
triggerName: string;
log: any;
context: Record<string, unknown>;
config: any;
}

export type BeforeSaveRequest<T extends ParseObject = ParseObject> = TriggerRequest<T>;
export type AfterSaveRequest<T extends ParseObject = ParseObject> = TriggerRequest<T>;
export type BeforeDeleteRequest<T extends ParseObject = ParseObject> = TriggerRequest<T>;
export type AfterDeleteRequest<T extends ParseObject = ParseObject> = TriggerRequest<T>;

export interface BeforeFindRequest<T extends ParseObject = ParseObject> {
installationId?: string;
master: boolean;
user?: ParseUser;
query: ParseQuery<T>;
ip: string;
headers: Record<string, string>;
triggerName: string;
log: any;
isGet: boolean;
config: any;
context: Record<string, unknown>;
readPreference?: string;
count?: boolean;
}

export interface AfterFindRequest<T extends ParseObject = ParseObject> {
installationId?: string;
master: boolean;
user?: ParseUser;
query: ParseQuery<T>;
results: T[];
ip: string;
headers: Record<string, string>;
triggerName: string;
log: any;
config: any;
context: Record<string, unknown>;
}

export interface FileTriggerRequest {
installationId?: string;
master: boolean;
user?: ParseUser;
file: ParseFile;
fileSize: number;
contentLength: number;
ip: string;
headers: Record<string, string>;
triggerName: string;
log: any;
config: any;
}

export interface ConnectTriggerRequest {
installationId?: string;
useMasterKey: boolean;
user?: ParseUser;
clients: number;
subscriptions: number;
sessionToken?: string;
}

export interface LiveQueryEventTrigger<T extends ParseObject = ParseObject> {
installationId?: string;
useMasterKey: boolean;
user?: ParseUser;
sessionToken?: string;
event: string;
object: T;
original?: T;
clients: number;
subscriptions: number;
sendEvent: boolean;
}

export interface JobRequest {
params: Record<string, any>;
message: (message: string) => void;
config: any;
}

// ============================================================================
// Validator Types
// ============================================================================

export interface ValidatorField {
type?: any;
constant?: boolean;
default?: any;
options?: any[] | (() => any[]) | any;
required?: boolean;
error?: string;
}

export interface ValidatorObject {
requireUser?: boolean;
requireMaster?: boolean;
validateMasterKey?: boolean;
skipWithMasterKey?: boolean;
requireAnyUserRoles?: string[] | (() => string[]);
requireAllUserRoles?: string[] | (() => string[]);
requireUserKeys?: string[] | Record<string, ValidatorField>;
fields?: string[] | Record<string, ValidatorField>;
rateLimit?: {
requestPath?: string;
requestMethods?: string | string[];
requestTimeWindow?: number;
requestCount?: number;
errorResponseMessage?: string;
includeInternalRequests?: boolean;
includeMasterKey?: boolean;
};
}

// ============================================================================
// HTTP Types
// ============================================================================

export interface HTTPOptions {
body?: string | object;
error?: (response: HTTPResponse) => void;
followRedirects?: boolean;
headers?: Record<string, string>;
method?: 'GET' | 'POST' | 'PUT' | 'DELETE' | 'HEAD' | 'OPTIONS';
params?: string | Record<string, string>;
success?: (response: HTTPResponse) => void;
url: string;
}

export interface HTTPResponse {
buffer?: Buffer;
cookies?: Record<string, any>;
data?: any;
headers?: Record<string, string>;
status: number;
text?: string;
}

// ============================================================================
// Enum
// ============================================================================

export enum ReadPreferenceOption {
Primary = 'PRIMARY',
PrimaryPreferred = 'PRIMARY_PREFERRED',
Secondary = 'SECONDARY',
SecondaryPreferred = 'SECONDARY_PREFERRED',
Nearest = 'NEAREST',
}

// ============================================================================
// Function Declarations (server-side only, implemented by Parse Server)
// ============================================================================

export declare function define<T extends Record<string, any> = Record<string, any>>(
name: string,
handler: (request: FunctionRequest<T>) => any,
validator?: ValidatorObject | ((request: FunctionRequest<T>) => any)
): void;

export declare function job(name: string, handler: (request: JobRequest) => any): void;

export declare function beforeSave<T extends ParseObject = ParseObject>(
className: string | (new () => T),
handler: (request: BeforeSaveRequest<T>) => T | undefined | Promise<T | undefined>,
validator?: ValidatorObject | ((request: BeforeSaveRequest<T>) => any)
): void;

export declare function afterSave<T extends ParseObject = ParseObject>(
className: string | (new () => T),
handler: (request: AfterSaveRequest<T>) => Promise<void> | undefined,
validator?: ValidatorObject | ((request: AfterSaveRequest<T>) => any)
): void;

export declare function beforeDelete<T extends ParseObject = ParseObject>(
className: string | (new () => T),
handler: (request: BeforeDeleteRequest<T>) => Promise<void> | undefined,
validator?: ValidatorObject | ((request: BeforeDeleteRequest<T>) => any)
): void;

export declare function afterDelete<T extends ParseObject = ParseObject>(
className: string | (new () => T),
handler: (request: AfterDeleteRequest<T>) => Promise<void> | undefined,
validator?: ValidatorObject | ((request: AfterDeleteRequest<T>) => any)
): void;

export declare function beforeFind<T extends ParseObject = ParseObject>(
className: string | (new () => T),
handler: (request: BeforeFindRequest<T>) => ParseQuery<T> | undefined | Promise<ParseQuery<T> | undefined>,
validator?: ValidatorObject | ((request: BeforeFindRequest<T>) => any)
): void;

export declare function afterFind<T extends ParseObject = ParseObject>(
className: string | (new () => T),
handler: (request: AfterFindRequest<T>) => T[] | undefined | Promise<T[] | undefined>,
validator?: ValidatorObject | ((request: AfterFindRequest<T>) => any)
): void;

export declare function beforeLogin(
handler: (request: TriggerRequest<ParseUser>) => Promise<void> | undefined,
validator?: ValidatorObject | ((request: TriggerRequest<ParseUser>) => any)
): void;

export declare function afterLogin(
handler: (request: TriggerRequest<ParseUser>) => Promise<void> | undefined
): void;

export declare function afterLogout(
handler: (request: TriggerRequest) => Promise<void> | undefined
): void;

export declare function beforePasswordResetRequest(
handler: (request: TriggerRequest<ParseUser>) => Promise<void> | undefined,
validator?: ValidatorObject | ((request: TriggerRequest<ParseUser>) => any)
): void;

export declare function beforeSaveFile(
handler: (request: FileTriggerRequest) => ParseFile | undefined | Promise<ParseFile | undefined>
): void;

export declare function afterSaveFile(
handler: (request: FileTriggerRequest) => Promise<void> | undefined
): void;

export declare function beforeDeleteFile(
handler: (request: FileTriggerRequest) => Promise<void> | undefined
): void;

export declare function afterDeleteFile(
handler: (request: FileTriggerRequest) => Promise<void> | undefined
): void;

export declare function beforeConnect(
handler: (request: ConnectTriggerRequest) => Promise<void> | undefined,
validator?: ValidatorObject | ((request: ConnectTriggerRequest) => any)
): void;

export declare function beforeSubscribe<T extends ParseObject = ParseObject>(
className: string | (new () => T),
handler: (request: TriggerRequest<T>) => Promise<void> | undefined,
validator?: ValidatorObject | ((request: TriggerRequest<T>) => any)
): void;

export declare function afterLiveQueryEvent<T extends ParseObject = ParseObject>(
className: string | (new () => T),
handler: (request: LiveQueryEventTrigger<T>) => Promise<void> | undefined,
validator?: ValidatorObject | ((request: LiveQueryEventTrigger<T>) => any)
): void;

export declare function sendEmail(data: {
from?: string;
to: string;
subject?: string;
text?: string;
html?: string;
}): Promise<void>;

export declare function httpRequest(options: HTTPOptions): Promise<HTTPResponse>;

// ============================================================================
// JSDoc Documentation (for documentation generation)
// ============================================================================
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you move the documentation to their approprate function and ensure jsdocs run and the docs are viewable?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  • Moved JSDoc documentation directly above each function declaration in CloudCode.ts
  • Verified npm run docs runs successfully without errors


/**
* Defines a Cloud Function.
*
Expand Down
Loading