Skip to content

Commit

Permalink
Merge pull request #10177 from hicommonwealth/salman/Issue#8711/dissc…
Browse files Browse the repository at this point in the history
…onnect-all-addresses

Disconnect all addresses
  • Loading branch information
salman-neslit authored Dec 18, 2024
2 parents 1412ae8 + af668f3 commit 75683cf
Show file tree
Hide file tree
Showing 4 changed files with 172 additions and 13 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,12 @@ type AddressProps = {
type AddressDetailsProps = {
profile: NewProfile;
addressInfo: AddressInfo;
toggleRemoveModal: (val: boolean, address: AddressInfo) => void;
toggleRemoveModal: (
val: boolean,
address: AddressInfo,
isBulkDelete: boolean,
communityName: string,
) => void;
};

type LinkedAddressesProps = {
Expand Down Expand Up @@ -54,6 +59,8 @@ const AddressDetails = (props: AddressDetailsProps) => {
enabled: !!community.id,
});

if (!fetchedCommunity) return null;

return (
<div className="AddressDetails">
<CWTruncatedAddress
Expand All @@ -67,7 +74,18 @@ const AddressDetails = (props: AddressDetailsProps) => {
menuItems={[
{
label: `Disconnect ${formatAddressShort(address)}`,
onClick: () => toggleRemoveModal(true, addressInfo),
onClick: () =>
toggleRemoveModal(
true,
addressInfo,
false,
fetchedCommunity.name,
),
},
{
label: 'Delete All Addresses',
onClick: () =>
toggleRemoveModal(true, addressInfo, true, fetchedCommunity.name),
},
]}
renderTrigger={(onclick) => (
Expand Down Expand Up @@ -98,6 +116,10 @@ export const LinkedAddresses = (props: LinkedAddressesProps) => {
const [currentAddress, setCurrentAddress] = useState<AddressInfo | null>(
null,
);
const [isBulkDeleteState, setIsBulkDeleteState] = useState(false);
const [selectedCommuinty, setSelectedCommunity] = useState<string | null>(
null,
);

const { profile, addresses, refreshProfiles } = props;

Expand All @@ -123,9 +145,13 @@ export const LinkedAddresses = (props: LinkedAddressesProps) => {
toggleRemoveModal={(
val: boolean,
selectedAddress: AddressInfo,
isBulkDelete: boolean = false,
community,
) => {
setIsRemoveModalOpen(val);
setCurrentAddress(selectedAddress);
setIsBulkDeleteState(isBulkDelete);
setSelectedCommunity(community);
}}
/>
);
Expand All @@ -147,7 +173,8 @@ export const LinkedAddresses = (props: LinkedAddressesProps) => {
<CWModal
size="small"
content={
currentAddress && (
currentAddress &&
selectedCommuinty && (
<DeleteAddressModal
addresses={addresses}
address={currentAddress}
Expand All @@ -156,12 +183,15 @@ export const LinkedAddresses = (props: LinkedAddressesProps) => {
setIsRemoveModalOpen(false);
refreshProfiles(currentAddress);
}}
isBulkDelete={isBulkDeleteState}
communityName={selectedCommuinty}
/>
)
}
onClose={() => {
setIsRemoveModalOpen(false);
setCurrentAddress(null);
setSelectedCommunity(null);
}}
open={isRemoveModalOpen}
/>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,13 +24,17 @@ type DeleteAddressModalAttrs = {
address: AddressInfo;
chain: string;
closeModal: () => void;
isBulkDelete?: boolean;
communityName: string;
};

export const DeleteAddressModal = ({
address,
addresses,
chain,
closeModal,
isBulkDelete = false,
communityName,
}: DeleteAddressModalAttrs) => {
const user = useUserStore();

Expand All @@ -42,11 +46,13 @@ export const DeleteAddressModal = ({
}

try {
const response = await axios.post(`${SERVER_URL}/deleteAddress`, {
address: address.address,
chain,
jwt: user.jwt,
});
const payload = { address: address?.address, chain, jwt: user.jwt };

const endpoint = isBulkDelete
? `${SERVER_URL}/deleteAllAddresses`
: `${SERVER_URL}/deleteAddress`;

const response = await axios.post(endpoint, payload);

if (response?.data.status === 'Success') {
const updatedAddresses = [...user.addresses].filter(
Expand Down Expand Up @@ -87,15 +93,21 @@ export const DeleteAddressModal = ({
return (
<div className="DeleteAddressModal">
<CWModalHeader
label={`Disconnect ${formatAddressShort(address.address)}`}
label={
isBulkDelete
? 'Disconnect All Addresses'
: `Disconnect ${formatAddressShort(address?.address || '')}`
}
icon="danger"
onModalClose={closeModal}
/>
<CWModalBody>
<CWText>
By removing this address you will be leaving the{' '}
{address.community.id}. Your contributions and comments will remain.
Don&apos;t worry, you can rejoin anytime.
{isBulkDelete
? `By leaving ${communityName} you will disconnect all
linked addresses. Your threads will remain intact.`
: `By removing this address you will be leaving the ${communityName}.
Your contributions and comments will remain. Don't worry, you can rejoin anytime.`}
</CWText>
</CWModalBody>
<CWModalFooter>
Expand All @@ -106,7 +118,7 @@ export const DeleteAddressModal = ({
buttonHeight="sm"
/>
<CWButton
label="Disconnect Address"
label={isBulkDelete ? 'Disconnect All' : 'Disconnect Address'}
buttonType="destructive"
onClick={handleDelete}
buttonHeight="sm"
Expand Down
108 changes: 108 additions & 0 deletions packages/commonwealth/server/routes/deleteAllAddress.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,108 @@
import { AppError } from '@hicommonwealth/core';
import type { DB } from '@hicommonwealth/model';
import { decrementProfileCount } from '@hicommonwealth/model';
import { WalletId } from '@hicommonwealth/shared';
import type { NextFunction, Request, Response } from 'express';

export const Errors = {
NotLoggedIn: 'Not signed in',
NeedAddress: 'Must provide address',
NeedCommunity: 'Must provide community',
AddressNotFound: 'Address not found',
CannotDeleteMagic: 'Cannot delete Magic Link address',
CannotDeleteOnlyAdmin:
'Community must have at least 1 admin. Please assign another community member as admin, to leave this community.',
};

const deleteAllAddress = async (
models: DB,
req: Request,
res: Response,
next: NextFunction,
) => {
const { community } = req;

if (!req.user) {
return next(new AppError(Errors.NotLoggedIn));
}
if (!req.body.address) {
return next(new AppError(Errors.NeedAddress));
}
if (!community) {
return next(new AppError(Errors.NeedCommunity));
}

const addressObj = await models.Address.findOne({
where: {
community_id: community.id,
address: req.body.address,
user_id: req.user.id,
},
});
if (!addressObj) {
return next(new AppError(Errors.AddressNotFound));
}
if (addressObj.wallet_id === WalletId.Magic) {
return next(new AppError(Errors.CannotDeleteMagic));
}

const [AssociatedAddresses, adminUsers] = await Promise.all([
models.Address.findAll({
where: {
user_id: addressObj.user_id,
community_id: addressObj.community_id,
},
}),
models.Address.findAll({
where: {
community_id: community.id,
role: 'admin',
},
}),
]);

if (
AssociatedAddresses.some(
(associatedAddress) =>
associatedAddress.dataValues.wallet_id === WalletId.Magic,
)
) {
return next(new AppError(Errors.CannotDeleteMagic));
}

if (
adminUsers.length === 1 &&
adminUsers.some((adminUser) =>
AssociatedAddresses.some(
(associatedAddress) =>
adminUser.dataValues.address === associatedAddress.dataValues.address,
),
)
) {
return next(new AppError(Errors.CannotDeleteOnlyAdmin));
}

await models.sequelize.transaction(async (transaction) => {
const associatedAddressIds = AssociatedAddresses.map(
(address) => address.dataValues.id,
).filter((id): id is number => id !== undefined);

if (associatedAddressIds.length > 0) {
await models.Address.update(
{ user_id: null, verified: null },
{
where: {
id: associatedAddressIds,
},
transaction,
},
);
}

await decrementProfileCount(community.id!, req.user!.id!, transaction);
});

return res.json({ status: 'Success', response: 'Deleted Address' });
};

export default deleteAllAddress;
9 changes: 9 additions & 0 deletions packages/commonwealth/server/routing/router.ts
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,7 @@ import { ServerTagsController } from 'server/controllers/server_tags_controller'
import { rateLimiterMiddleware } from 'server/middleware/rateLimiter';
import { getTopUsersHandler } from 'server/routes/admin/get_top_users_handler';
import { getNamespaceMetadata } from 'server/routes/communities/get_namespace_metadata';
import deleteAllAddress from 'server/routes/deleteAllAddress';
import { config } from '../config';
import { getStatsHandler } from '../routes/admin/get_stats_handler';
import { getCanvasClockHandler } from '../routes/canvas/get_canvas_clock_handler';
Expand Down Expand Up @@ -176,6 +177,14 @@ function setupRouter(
databaseValidationService.validateCommunity,
deleteAddress.bind(this, models),
);
registerRoute(
router,
'post',
'/deleteAllAddresses',
passport.authenticate('jwt', { session: false }),
databaseValidationService.validateCommunity,
deleteAllAddress.bind(this, models),
);
registerRoute(
router,
'post',
Expand Down

0 comments on commit 75683cf

Please sign in to comment.