Skip to content

Commit

Permalink
Merge pull request #543 from gitroomhq/feat/short-linking
Browse files Browse the repository at this point in the history
Added shortlinking option
  • Loading branch information
nevo-david authored Jan 12, 2025
2 parents 9f6a211 + 772ff4c commit d2ba248
Show file tree
Hide file tree
Showing 12 changed files with 480 additions and 27 deletions.
2 changes: 2 additions & 0 deletions apps/backend/src/api/api.module.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ import { AgenciesController } from '@gitroom/backend/api/routes/agencies.control
import { PublicController } from '@gitroom/backend/api/routes/public.controller';
import { RootController } from '@gitroom/backend/api/routes/root.controller';
import { TrackService } from '@gitroom/nestjs-libraries/track/track.service';
import { ShortLinkService } from '@gitroom/nestjs-libraries/short-linking/short.link.service';

const authenticatedController = [
UsersController,
Expand Down Expand Up @@ -63,6 +64,7 @@ const authenticatedController = [
CodesService,
IntegrationManager,
TrackService,
ShortLinkService,
],
get exports() {
return [...this.imports, ...this.providers];
Expand Down
38 changes: 20 additions & 18 deletions apps/backend/src/api/routes/posts.controller.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ import { CreateGeneratedPostsDto } from '@gitroom/nestjs-libraries/dtos/generato
import { AgentGraphService } from '@gitroom/nestjs-libraries/agent/agent.graph.service';
import { Response } from 'express';
import { GetUserFromRequest } from '@gitroom/nestjs-libraries/user/user.from.request';
import { ShortLinkService } from '@gitroom/nestjs-libraries/short-linking/short.link.service';

@ApiTags('Posts')
@Controller('/posts')
Expand All @@ -35,9 +36,23 @@ export class PostsController {
private _postsService: PostsService,
private _starsService: StarsService,
private _messagesService: MessagesService,
private _agentGraphService: AgentGraphService
private _agentGraphService: AgentGraphService,
private _shortLinkService: ShortLinkService
) {}

@Get('/:id/statistics')
async getStatistics(
@GetOrgFromRequest() org: Organization,
@Param('id') id: string
) {
return this._postsService.getStatistics(org.id, id);
}

@Post('/should-shortlink')
async shouldShortlink(@Body() body: { messages: string[] }) {
return { ask: this._shortLinkService.askShortLinkedin(body.messages) };
}

@Get('/marketplace/:id?')
async getMarketplacePosts(
@GetOrgFromRequest() org: Organization,
Expand All @@ -61,26 +76,16 @@ export class PostsController {
@GetOrgFromRequest() org: Organization,
@Query() query: GetPostsDto
) {
const [posts] = await Promise.all([
this._postsService.getPosts(org.id, query),
// this._commentsService.getAllCommentsByWeekYear(
// org.id,
// query.year,
// query.week
// ),
]);
const posts = await this._postsService.getPosts(org.id, query);

return {
posts,
// comments,
};
}

@Get('/find-slot')
async findSlot(
@GetOrgFromRequest() org: Organization,
) {
return {date: await this._postsService.findFreeDateTime(org.id)}
async findSlot(@GetOrgFromRequest() org: Organization) {
return { date: await this._postsService.findFreeDateTime(org.id) };
}

@Get('/predict-trending')
Expand Down Expand Up @@ -128,10 +133,7 @@ export class PostsController {
@Res({ passthrough: false }) res: Response
) {
res.setHeader('Content-Type', 'application/json; charset=utf-8');
for await (const event of this._agentGraphService.start(
org.id,
body,
)) {
for await (const event of this._agentGraphService.start(org.id, body)) {
res.write(JSON.stringify(event) + '\n');
}

Expand Down
23 changes: 21 additions & 2 deletions apps/frontend/src/components/launches/add.edit.model.tsx
Original file line number Diff line number Diff line change
@@ -1,14 +1,12 @@
'use client';

import React, {
ClipboardEventHandler,
FC,
Fragment,
MouseEventHandler,
useCallback,
useEffect,
useMemo,
useRef,
ClipboardEvent,
useState,
memo,
Expand Down Expand Up @@ -363,12 +361,33 @@ export const AddEditModal: FC<{
}
}

const shortLinkUrl = await (
await fetch('/posts/should-shortlink', {
method: 'POST',
body: JSON.stringify({
messages: allKeys.flatMap((p) =>
p.value.flatMap((a) =>
a.content.slice(0, p.maximumCharacters || 1000000)
)
),
}),
})
).json();

const shortLink = !shortLinkUrl.ask
? false
: await deleteDialog(
'Do you want to shortlink the URLs? it will let you get statistics over clicks',
'Yes, shortlink it!'
);

setLoading(true);
await fetch('/posts', {
method: 'POST',
body: JSON.stringify({
...(postFor ? { order: postFor.id } : {}),
type,
shortLink,
date: dateState.utc().format('YYYY-MM-DDTHH:mm:ss'),
posts: allKeys.map((p) => ({
...p,
Expand Down
101 changes: 96 additions & 5 deletions apps/frontend/src/components/launches/calendar.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,7 @@ import { extend } from 'dayjs';
import { isUSCitizen } from './helpers/isuscitizen.utils';
import removeMd from 'remove-markdown';
import { useInterval } from '@mantine/hooks';
import { StatisticsModal } from '@gitroom/frontend/components/launches/statistics';
extend(isSameOrAfter);
extend(isSameOrBefore);

Expand Down Expand Up @@ -508,6 +509,23 @@ export const CalendarColumn: FC<{
});
}, [integrations, getDate]);

const openStatistics = useCallback(
(id: string) => () => {
modal.openModal({
closeOnClickOutside: true,
closeOnEscape: true,
withCloseButton: false,
classNames: {
modal: 'w-[100%] max-w-[1400px] bg-transparent text-textColor',
},
children: <StatisticsModal postId={id} />,
size: '80%',
// title: `Adding posts for ${getDate.format('DD/MM/YYYY HH:mm')}`,
});
},
[]
);

const addProvider = useAddProvider();

return (
Expand Down Expand Up @@ -551,6 +569,7 @@ export const CalendarColumn: FC<{
isBeforeNow={isBeforeNow}
date={getDate}
state={post.state}
statistics={openStatistics(post.id)}
editPost={editPost(post, false)}
duplicatePost={editPost(post, true)}
post={post}
Expand Down Expand Up @@ -654,13 +673,22 @@ const CalendarItem: FC<{
isBeforeNow: boolean;
editPost: () => void;
duplicatePost: () => void;
statistics: () => void;
integrations: Integrations[];
state: State;
display: 'day' | 'week' | 'month';
post: Post & { integration: Integration };
}> = memo((props) => {
const { editPost, duplicatePost, post, date, isBeforeNow, state, display } =
props;
const {
editPost,
statistics,
duplicatePost,
post,
date,
isBeforeNow,
state,
display,
} = props;

const preview = useCallback(() => {
window.open(`/p/` + post.id + '?share=true', '_blank');
Expand All @@ -683,18 +711,24 @@ const CalendarItem: FC<{
className={clsx('w-full flex h-full flex-1 flex-col group', 'relative')}
style={{ opacity }}
>
<div className="text-primary bg-forth text-[11px] h-[15px] w-full rounded-tr-[10px] rounded-tl-[10px] flex justify-center gap-[10px] px-[5px]">
<div className="text-white bg-forth text-[11px] h-[15px] w-full rounded-tr-[10px] rounded-tl-[10px] flex justify-center gap-[10px] px-[5px]">
<div
className="hidden group-hover:block hover:underline cursor-pointer"
onClick={duplicatePost}
>
Duplicate
<Duplicate />
</div>
<div
className="hidden group-hover:block hover:underline cursor-pointer"
onClick={preview}
>
Preview
<Preview />
</div>{' '}
<div
className="hidden group-hover:block hover:underline cursor-pointer"
onClick={statistics}
>
<Statistics />
</div>
</div>
<div
Expand Down Expand Up @@ -730,3 +764,60 @@ const CalendarItem: FC<{
</div>
);
});

const Duplicate = () => {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
width="15"
height="15"
viewBox="0 0 32 32"
fill="none"
data-tooltip-id="tooltip"
data-tooltip-content="Duplicate Post"
>
<path
d="M27 5H9C8.46957 5 7.96086 5.21071 7.58579 5.58579C7.21071 5.96086 7 6.46957 7 7V9H5C4.46957 9 3.96086 9.21071 3.58579 9.58579C3.21071 9.96086 3 10.4696 3 11V25C3 25.5304 3.21071 26.0391 3.58579 26.4142C3.96086 26.7893 4.46957 27 5 27H23C23.5304 27 24.0391 26.7893 24.4142 26.4142C24.7893 26.0391 25 25.5304 25 25V23H27C27.5304 23 28.0391 22.7893 28.4142 22.4142C28.7893 22.0391 29 21.5304 29 21V7C29 6.46957 28.7893 5.96086 28.4142 5.58579C28.0391 5.21071 27.5304 5 27 5ZM23 11V13H5V11H23ZM23 25H5V15H23V25ZM27 21H25V11C25 10.4696 24.7893 9.96086 24.4142 9.58579C24.0391 9.21071 23.5304 9 23 9H9V7H27V21Z"
fill="white"
/>
</svg>
);
};

const Preview = () => {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
width="15"
height="15"
viewBox="0 0 32 32"
fill="none"
data-tooltip-id="tooltip"
data-tooltip-content="Preview Post"
>
<path
d="M30.9137 15.595C30.87 15.4963 29.8112 13.1475 27.4575 10.7937C24.3212 7.6575 20.36 6 16 6C11.64 6 7.67874 7.6575 4.54249 10.7937C2.18874 13.1475 1.12499 15.5 1.08624 15.595C1.02938 15.7229 1 15.8613 1 16.0012C1 16.1412 1.02938 16.2796 1.08624 16.4075C1.12999 16.5062 2.18874 18.8538 4.54249 21.2075C7.67874 24.3425 11.64 26 16 26C20.36 26 24.3212 24.3425 27.4575 21.2075C29.8112 18.8538 30.87 16.5062 30.9137 16.4075C30.9706 16.2796 31 16.1412 31 16.0012C31 15.8613 30.9706 15.7229 30.9137 15.595ZM16 24C12.1525 24 8.79124 22.6012 6.00874 19.8438C4.86704 18.7084 3.89572 17.4137 3.12499 16C3.89551 14.5862 4.86686 13.2915 6.00874 12.1562C8.79124 9.39875 12.1525 8 16 8C19.8475 8 23.2087 9.39875 25.9912 12.1562C27.1352 13.2912 28.1086 14.5859 28.8812 16C27.98 17.6825 24.0537 24 16 24ZM16 10C14.8133 10 13.6533 10.3519 12.6666 11.0112C11.6799 11.6705 10.9108 12.6075 10.4567 13.7039C10.0026 14.8003 9.88377 16.0067 10.1153 17.1705C10.3468 18.3344 10.9182 19.4035 11.7573 20.2426C12.5965 21.0818 13.6656 21.6532 14.8294 21.8847C15.9933 22.1162 17.1997 21.9974 18.2961 21.5433C19.3924 21.0892 20.3295 20.3201 20.9888 19.3334C21.6481 18.3467 22 17.1867 22 16C21.9983 14.4092 21.3657 12.884 20.2408 11.7592C19.1159 10.6343 17.5908 10.0017 16 10ZM16 20C15.2089 20 14.4355 19.7654 13.7777 19.3259C13.1199 18.8864 12.6072 18.2616 12.3045 17.5307C12.0017 16.7998 11.9225 15.9956 12.0768 15.2196C12.2312 14.4437 12.6122 13.731 13.1716 13.1716C13.731 12.6122 14.4437 12.2312 15.2196 12.0769C15.9956 11.9225 16.7998 12.0017 17.5307 12.3045C18.2616 12.6072 18.8863 13.1199 19.3259 13.7777C19.7654 14.4355 20 15.2089 20 16C20 17.0609 19.5786 18.0783 18.8284 18.8284C18.0783 19.5786 17.0609 20 16 20Z"
fill="white"
/>
</svg>
);
};

export const Statistics = () => {
return (
<svg
xmlns="http://www.w3.org/2000/svg"
width="15"
height="15"
viewBox="0 0 32 32"
fill="none"
data-tooltip-id="tooltip"
data-tooltip-content="Post Statistics"
>
<path
d="M28 25H27V5C27 4.73478 26.8946 4.48043 26.7071 4.29289C26.5196 4.10536 26.2652 4 26 4H19C18.7348 4 18.4804 4.10536 18.2929 4.29289C18.1054 4.48043 18 4.73478 18 5V10H12C11.7348 10 11.4804 10.1054 11.2929 10.2929C11.1054 10.4804 11 10.7348 11 11V16H6C5.73478 16 5.48043 16.1054 5.29289 16.2929C5.10536 16.4804 5 16.7348 5 17V25H4C3.73478 25 3.48043 25.1054 3.29289 25.2929C3.10536 25.4804 3 25.7348 3 26C3 26.2652 3.10536 26.5196 3.29289 26.7071C3.48043 26.8946 3.73478 27 4 27H28C28.2652 27 28.5196 26.8946 28.7071 26.7071C28.8946 26.5196 29 26.2652 29 26C29 25.7348 28.8946 25.4804 28.7071 25.2929C28.5196 25.1054 28.2652 25 28 25ZM20 6H25V25H20V6ZM13 12H18V25H13V12ZM7 18H11V25H7V18Z"
fill="white"
/>
</svg>
);
};
73 changes: 73 additions & 0 deletions apps/frontend/src/components/launches/statistics.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
import React, { FC, Fragment, useCallback } from 'react';
import { useModals } from '@mantine/modals';
import useSWR from 'swr';
import { useFetch } from '@gitroom/helpers/utils/custom.fetch';

export const StatisticsModal: FC<{ postId: string }> = (props) => {
const { postId } = props;
const modals = useModals();
const fetch = useFetch();

const loadStatistics = useCallback(async () => {
return (await fetch(`/posts/${postId}/statistics`)).json();
}, [postId]);

const closeAll = useCallback(() => {
modals.closeAll();
}, []);

const { data, isLoading } = useSWR(
`/posts/${postId}/statistics`,
loadStatistics
);

return (
<div className="bg-sixth p-[32px] w-full max-w-[920px] mx-auto flex flex-col rounded-[4px] border border-customColor6 relative">
<button
onClick={closeAll}
className="outline-none absolute right-[20px] top-[15px] mantine-UnstyledButton-root mantine-ActionIcon-root hover:bg-tableBorder cursor-pointer mantine-Modal-close mantine-1dcetaa"
type="button"
>
<svg
viewBox="0 0 15 15"
fill="none"
xmlns="http://www.w3.org/2000/svg"
width="16"
height="16"
>
<path
d="M11.7816 4.03157C12.0062 3.80702 12.0062 3.44295 11.7816 3.2184C11.5571 2.99385 11.193 2.99385 10.9685 3.2184L7.50005 6.68682L4.03164 3.2184C3.80708 2.99385 3.44301 2.99385 3.21846 3.2184C2.99391 3.44295 2.99391 3.80702 3.21846 4.03157L6.68688 7.49999L3.21846 10.9684C2.99391 11.193 2.99391 11.557 3.21846 11.7816C3.44301 12.0061 3.80708 12.0061 4.03164 11.7816L7.50005 8.31316L10.9685 11.7816C11.193 12.0061 11.5571 12.0061 11.7816 11.7816C12.0062 11.557 12.0062 11.193 11.7816 10.9684L8.31322 7.49999L11.7816 4.03157Z"
fill="currentColor"
fillRule="evenodd"
clipRule="evenodd"
/>
</svg>
</button>
<h1 className="text-[24px]">Statistics</h1>
{isLoading ? (
<div>Loading</div>
) : (
<>
{data.clicks.length === 0 ? (
'No Results'
) : (
<>
<div className="grid grid-cols-3 mt-[20px]">
<div className="bg-forth p-[4px] rounded-tl-lg">Short Link</div>
<div className="bg-forth p-[4px]">Original Link</div>
<div className="bg-forth p-[4px] rounded-tr-lg">Clicks</div>
{data.clicks.map((p: any) => (
<Fragment key={p.short}>
<div className="p-[4px] py-[10px] bg-customColor6">{p.short}</div>
<div className="p-[4px] py-[10px] bg-customColor6">{p.original}</div>
<div className="p-[4px] py-[10px] bg-customColor6">{p.clicks}</div>
</Fragment>
))}
</div>
</>
)}
</>
)}
</div>
);
};
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ import { OpenaiService } from '@gitroom/nestjs-libraries/openai/openai.service';
import { AgenciesService } from '@gitroom/nestjs-libraries/database/prisma/agencies/agencies.service';
import { AgenciesRepository } from '@gitroom/nestjs-libraries/database/prisma/agencies/agencies.repository';
import { TrackService } from '@gitroom/nestjs-libraries/track/track.service';
import { ShortLinkService } from '@gitroom/nestjs-libraries/short-linking/short.link.service';

@Global()
@Module({
Expand Down Expand Up @@ -64,6 +65,7 @@ import { TrackService } from '@gitroom/nestjs-libraries/track/track.service';
OpenaiService,
EmailService,
TrackService,
ShortLinkService,
],
get exports() {
return this.providers;
Expand Down
Loading

0 comments on commit d2ba248

Please sign in to comment.