-
-
Save chaodonghu/c942b6ca8f8c247ccacd3b0123ff3580 to your computer and use it in GitHub Desktop.
// Run GOOGLE CHROME - WORKING AS OF MARCH 23 2025 | |
// Please @ me in the comments if this stops working, I will try to get it working again within the month | |
// INSTRUCTIONS | |
// 1. Open Instagram in Chrome | |
// 2. Click on "FOLLOWING" on your Instagram profile | |
// 3. Open developer tools by right clicking on the page and clicking "INSPECT" | |
// 4. Copy the code below and paste in the developer tools console and press enter to run | |
// 5. Script will not run if tab is navigated away from, minimized of unfocused (It is recommended to open a new chrome window or push tab to the side and let script run in background) | |
const unfollowEveryone = (async () => { | |
// Modify these variables to your liking | |
// UNFOLLOW_LIMIT is the number of users to unfollow (this is set to 1000) | |
const UNFOLLOW_LIMIT = 1000; | |
const BREAK_DURATION = 5 * 60 * 1000; // 5 minutes break | |
const TOTAL_DURATION = 10 * 60 * 1000; // 10 minutes duration - Timeout after 10 minutes | |
const delay = (ms) => new Promise((resolve) => setTimeout(resolve, ms)); | |
// Find target button | |
const findButtonByText = (text) => | |
Array.from(document.querySelectorAll("button")).find( | |
(button) => button.innerText === text | |
); | |
console.log("Start unfollowing script..."); | |
let startTime = new Date().getTime(); | |
// Track the number of unfollows | |
let unfollowCount = 0; | |
while (new Date().getTime() - startTime < TOTAL_DURATION) { | |
for (let i = 0; i < UNFOLLOW_LIMIT; i++) { | |
// UNFOLLOW_INTERVAL is the interval between unfollows | |
// Increase UNFOLLOW_INTERVAL if you are getting rate limited | |
// Set this to 0 unfollow as quickly as possible - not recommended | |
// Random unfollow interval for each follow to avoid rate limiting minimum 5 seconds | |
// This gets reset to a random number every loop | |
const UNFOLLOW_INTERVAL = Math.floor(Math.random() * 10 + 1) * 5000; | |
const followingButton = findButtonByText("Following"); | |
if (!followingButton) { | |
console.log("No more users to unfollow or unable to find button."); | |
break; | |
} | |
followingButton.scrollIntoViewIfNeeded(); | |
await followingButton.click(); | |
await delay(100); // Short delay to ensure button is clicked | |
const confirmUnfollowButton = findButtonByText("Unfollow"); | |
if (confirmUnfollowButton) { | |
await confirmUnfollowButton.click(); // Wait for the unfollow to complete | |
unfollowCount++; | |
console.log(`Unfollowed #${unfollowCount}`); | |
} else { | |
console.log("No unfollow confirmation button found."); | |
} | |
console.log(`Wait ${UNFOLLOW_INTERVAL} milliseconds`); | |
await delay(UNFOLLOW_INTERVAL); | |
} | |
console.log(`Taking a break for ${BREAK_DURATION / 1000 / 60} minutes...`); | |
await delay(BREAK_DURATION); // Take a break to avoid rate limiting | |
startTime = new Date().getTime(); // Reset start time for the next cycle | |
} | |
console.log("Unfollow script complete!"); | |
})(); |
const FOLLOWING_BUTTON_TEXT = 'Following'; // Change to match your localized Instagram version
const UNFOLLOW_BUTTON_TEXT = 'Unfollow'; // Change to match your localized Instagram version
const MAX_ATTEMPTS_PER_UNFOLLOW = 3; // Maximum number of attempts to find the Unfollow button
const MAX_TOTAL_ATTEMPTS = 200; // Maximum total attempts before exiting
const TEXT_XPATH = './/div//text()';
const TEXT_XPATH_UNFOLLOW = 'text()';
// Function to click on the Unfollow button
const unfollowUser = async () => {
const followingButton = document.evaluate(
`//button[${TEXT_XPATH}="${FOLLOWING_BUTTON_TEXT}"]`,
document,
null,
XPathResult.FIRST_ORDERED_NODE_TYPE,
null
).singleNodeValue;
if (!followingButton) {
console.log('Following button not found. Exiting...');
return true; // No following button found, script should stop
}
console.log('Found following button. Clicking...');
followingButton.click();
console.log('Clicked following button.');
// Wait for a brief moment
await timeout(2000);
let unfollowButton = null;
let attempts = 1;
while (attempts <= MAX_ATTEMPTS_PER_UNFOLLOW && !unfollowButton) {
console.log(`Attempt ${attempts}: Trying to find Unfollow button...`);
unfollowButton = document.evaluate(
`//button[${TEXT_XPATH_UNFOLLOW}="${UNFOLLOW_BUTTON_TEXT}"]`,
document,
null,
XPathResult.FIRST_ORDERED_NODE_TYPE,
null
).singleNodeValue;
attempts++;
await timeout(1000); // Wait for 1 second between attempts
}
if (unfollowButton) {
console.log('Found Unfollow button. Clicking...');
unfollowButton.click();
// Scroll down to force the next set of users to load
const scrollContainer = document.querySelector('[class="_aano"]');
if (scrollContainer) {
scrollContainer.scrollBy(0, 30);
}
} else {
console.log('Unfollow button not found after retries.');
}
return false; // Continue to the next user
};
// Function to add a delay between actions
const timeout = (ms) => new Promise(resolve => setTimeout(resolve, ms));
// Function to control the main unfollow loop
const unfollowEveryone = async () => {
let totalAttempts = 0;
while (totalAttempts < MAX_TOTAL_ATTEMPTS) {
totalAttempts++;
console.log(`Processing user #${totalAttempts}...`);
const shouldStop = await unfollowUser();
if (shouldStop) {
console.log('No more following buttons found. Exiting...');
break; // No more following buttons found, exit the loop
}
const unfollowTimeout = randomTimeout();
console.log(`Waiting ${unfollowTimeout / 1000} seconds before the next user.`);
await timeout(unfollowTimeout);
}
console.log('Done processing.');
};
// Function to generate a random timeout between 60-70 seconds
const randomTimeout = () => (Math.floor((Math.random() * 10) + 1) * 1000) + 60000;
// Start the unfollow process
unfollowEveryone();
This script should be working as of Dec 26 2023. Thanks @DimVat1 for that piece of code, i've added a wait interval and log to your code and added some comments/instructions for users.
Hope this helps!
The script works but only for the 20 or so accounts displayed. It is not able to load more accounts after that. A manual reload of the page is required to refresh the list of accounts.
The script works but only for the 20 or so accounts displayed. It is not able to load more accounts after that. A manual reload of the page is required to refresh the list of accounts.
@superbarney There is a TOTAL_DURATION
on line 18 that defines how long the script runs, by default I have it set at 10 minutes so the script will only run for 10 minutes. This in combination with the UNFOLLOW_INTERVAL
on line 48 which is by default a multiple of 5 seconds. Either increase the TOTAL_DURATION
or decrease the UNFOLLOW_INTERVAL
or play with the UNFOLLOW_LIMIT
and BREAK_DURATION
constants.
I've confirmed it to be still working as intended, a manual reload of the page isn't needed -> https://cdn.zappy.app/883b402c34b4343d3280c20730ee36fa.mp4
@Alaryne Yes both the "Following" modal and the "Followers" modal should work with this script if you're on an account page that is not yours.
If you mean removing followers that follow your account, I haven't tried it but you can modify line 33 and line 40 to find the "Remove" button instead.
@superbarney There is a
TOTAL_DURATION
on line 18 that defines how long the script runs, by default I have it set at 10 minutes so the script will only run for 10 minutes. This in combination with theUNFOLLOW_INTERVAL
on line 48 which is by default a multiple of 5 seconds. Either increase theTOTAL_DURATION
or decrease theUNFOLLOW_INTERVAL
or play with theUNFOLLOW_LIMIT
andBREAK_DURATION
constants.I've confirmed it to be still working as intended, a manual reload of the page isn't needed -> https://cdn.zappy.app/883b402c34b4343d3280c20730ee36fa.mp4
Thanks. I tried again and it did work as intended, although strangely, not consistently. Sometimes, it reaches a point and displays the message "Waiting 300 seconds ..." and it resumes after a period of time, but many times, it hangs at that stage.
I have managed to unfollow close to 3000 accounts with the script but had to do it over the course of about a week. To others seeking to do the same, I'd be patient and not mess with the default values as you will hit the rate limit much faster and be locked out from further reducing followers.
@chaodonghu Thanks for the script, much appreciated.
I can verify that the code operates as intended, but it encounters timeouts. Below is an enhanced version of the code, with a particular focus on respecting Instagram's rate limits—both unofficial and official—to mitigate spam and abuse, and improving error handling.
(async function () {
const UNFOLLOW_LIMIT = 200; // Reduced to stay under potential rate limits
const UNFOLLOW_INTERVAL = 5000; // Increased interval to be more respectful of rate limits
const BREAK_DURATION = 5 * 60 * 1000; // 5 minutes break
const TOTAL_DURATION = 10 * 60 * 1000; // 10 minutes duration
const delay = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
const findButtonByText = (text) =>
Array.from(document.querySelectorAll("button"))
.find((button) => button.innerText === text);
console.log("Start");
let startTime = Date.now();
let unfollowCount = 0;
while (Date.now() - startTime < TOTAL_DURATION) {
for (let i = 0; i < UNFOLLOW_LIMIT; i++) {
const followButton = findButtonByText("Following");
if (!followButton) {
console.log("No more users to unfollow or unable to find button.");
break;
}
followButton.scrollIntoViewIfNeeded();
followButton.click();
await delay(100); // Short delay to wait for modal
const confirmButton = findButtonByText("Unfollow");
if (confirmButton) {
await confirmButton.click(); // Confirm unfollow
unfollowCount++;
console.log(`Unfollowed #${unfollowCount}`);
} else {
console.log("Confirmation button not found.");
}
await delay(UNFOLLOW_INTERVAL);
}
console.log(`Taking a break for ${BREAK_DURATION / 1000} seconds...`);
await delay(BREAK_DURATION);
startTime = Date.now(); // Reset start time for the next cycle
}
console.log("The end");
})();
- Reduced UNFOLLOW_LIMIT and Increased UNFOLLOW_INTERVAL: These changes help to stay under Instagram's rate limits.
Improved Button Selection: Uses Array.from() for better readability and directly finds the button instead of mapping and then filtering. - Added Basic Logging: Basic logging for when no more follow buttons are found or the confirmation button isn't found, which helps in understanding the script's state.
- Error Handling and Debugging: While explicit error handling isn't added, logging helps identify issues. For actual production scripts, consider try-catch blocks around actions that might fail.
may be a silly question, but how do you stop the code from running?
may be a silly question, but how do you stop the code from running?
@pauldgayle You can refresh the page or manually set a limit to stop it from running after a certain limit as mentioned in the script.
// Modify these variables to your liking
const UNFOLLOW_LIMIT = 800;
const BREAK_DURATION = 5 * 60 * 1000; // 5 minutes break
const TOTAL_DURATION = 10 * 60 * 1000; // 10 minutes duration - Timeout after 10 minutes
thank you!!
is there an easy way to add a list of users I don’t want to unfollow?
is there an easy way to add a list of users I don’t want to unfollow?
@nikko-guy Unfortunately not since the entire script is based on just finding the "Following" button irrespective of the user you are unfollowing.
If you want to add a list of users you don't want to unfollow that would involve finding the "Following" button then the name/username element beside the "Following" button and skipping the iteration if it matches a name/username in some sort of lookup array or object.
@isalmanhaider Love your version .. but i saw "Taking a break for 300 seconds" and i just smiled a little then checked the script
provided a quick fix
console.log(`Taking a break for ${(BREAK_DURATION / 1000) / 60} minutes...`);
my account got restricted for "certain activity" after 400+ unfollows .. I will try again later tonight by modifying the script to have an interval before clicking the confirmation box and drop it here
oh its just
await delay(UNFOLLOW_INTERVAL);
confirmButton.click(); // Confirm unfollow
did they update the text to "Remove" instead of "Unfollow"? How to fix?
Also sometimes mine typically only does 15-20 and then stops as if iisn't scrolling, thoughts on how to fix?
My full fix to the script . 500+ unfollowed so far I guess it's good ...
const unfollowEveryone = (async function () {
const UNFOLLOW_LIMIT = 100; // Reduced to stay under potential rate limits
const UNFOLLOW_INTERVAL = 5000; // Increased interval to be more respectful of rate limits
const UNFOLLOW_INTERVAL_TWO = 2500; // You can increase this to 5000
const BREAK_DURATION = 3 * 60 * 1000; // 3 minutes break
const TOTAL_DURATION = 10 * 60 * 1000; // 10 minutes duration
const delay = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
const findButtonByText = (text) =>
Array.from(document.querySelectorAll("button"))
.find((button) => button.innerText === text);
console.log("Start");
let startTime = Date.now();
let unfollowCount = 0;
while (Date.now() - startTime < TOTAL_DURATION) {
for (let i = 0; i < UNFOLLOW_LIMIT; i++) {
const followButton = findButtonByText("Following");
if (!followButton) {
console.log("No more users to unfollow or unable to find button.");
break;
}
followButton.scrollIntoViewIfNeeded();
await followButton.click();
await delay(100); // Short delay to wait for modal
const confirmButton = findButtonByText("Unfollow");
if (confirmButton) {
await delay(UNFOLLOW_INTERVAL_TWO);
await confirmButton.click(); // Confirm unfollow
unfollowCount++;
console.log(`Unfollowed #${unfollowCount}`);
} else {
console.log("Confirmation button not found.");
}
await delay(UNFOLLOW_INTERVAL);
}
console.log(`Taking a break for ${(BREAK_DURATION / 1000) / 60} minutes...`);
await delay(BREAK_DURATION);
startTime = Date.now(); // Reset start time for the next cycle
}
console.log("The end");
})();
did they update the text to "Remove" instead of "Unfollow"? How to fix?
Also sometimes mine typically only does 15-20 and then stops as if iisn't scrolling, thoughts on how to fix?
@ghostronin try my fix . i included a delay for the unfollow button
My full fix to the script . 500+ unfollowed so far I guess it's good ...
const unfollowEveryone = (async function () { const UNFOLLOW_LIMIT = 100; // Reduced to stay under potential rate limits const UNFOLLOW_INTERVAL = 5000; // Increased interval to be more respectful of rate limits const UNFOLLOW_INTERVAL_TWO = 2500; // You can increase this to 5000 const BREAK_DURATION = 3 * 60 * 1000; // 3 minutes break const TOTAL_DURATION = 10 * 60 * 1000; // 10 minutes duration const delay = (ms) => new Promise((resolve) => setTimeout(resolve, ms)); const findButtonByText = (text) => Array.from(document.querySelectorAll("button")) .find((button) => button.innerText === text); console.log("Start"); let startTime = Date.now(); let unfollowCount = 0; while (Date.now() - startTime < TOTAL_DURATION) { for (let i = 0; i < UNFOLLOW_LIMIT; i++) { const followButton = findButtonByText("Following"); if (!followButton) { console.log("No more users to unfollow or unable to find button."); break; } followButton.scrollIntoViewIfNeeded(); await followButton.click(); await delay(100); // Short delay to wait for modal const confirmButton = findButtonByText("Unfollow"); if (confirmButton) { await delay(UNFOLLOW_INTERVAL_TWO); await confirmButton.click(); // Confirm unfollow unfollowCount++; console.log(`Unfollowed #${unfollowCount}`); } else { console.log("Confirmation button not found."); } await delay(UNFOLLOW_INTERVAL); } console.log(`Taking a break for ${(BREAK_DURATION / 1000) / 60} minutes...`); await delay(BREAK_DURATION); startTime = Date.now(); // Reset start time for the next cycle } console.log("The end"); })();
800+ unfollowed so far without restrictions .. best mod for 2025 👌
UPDATE: Works perfect, thank you so much!
@drekzikko Thank you! I'll try now.
Thanks @drekzikko, i've updated the script so it is working as of March 23 2025. I don't think you want the UNFOLLOW_INTERVAL
to be constant as you have it set in line 2 and 3
const UNFOLLOW_INTERVAL = 5000; // Increased interval to be more respectful of rate limits
const UNFOLLOW_INTERVAL_TWO = 2500; // You can increase this to 5000
You'll run into rate limiting/bot detection since the time between unfollows will be constant, I have it inside the loop so the interval is randomized by a multiple of 5ms everytime to avoid bot detecting so the clicks are more random.
@chaodonghu good idea .. love the update but you should squeeze in the second interval (delay actually) .that was for the confirmation button .. it clicks fast and that would be fishy right.
-randomize by 2500ms
can I get the script for removing followers instead of following
can I get the script for removing followers instead of following
@rashadraz you wanna remove all ur followers? wow
@chaodonghu good idea .. love the update but you should squeeze in the second interval (delay actually) .that was for the confirmation button .. it clicks fast and that would be fishy right.
-randomize by 2.5ms
@drekzikko Yeah, I think the second delay will double the script's runtime and might be unnecessary since we’re already waiting for the button to be found. Plus, the bot detection happens at the request level, and we already have a delay right after the button is clicked.
can I get the script for removing followers instead of following
@rashadraz Hey, you should be able to adjust the script to find the "Followers" link and then the "Remove" button, but I don’t see a strong use case for this. So, I won’t be creating a script for it. The goal of the three scripts (like, follow and unfollow) was to increase traction and visibility for your account, and removing followers seems to go against that purpose.
@drekzikko Yeah, I think the second delay will double the script's runtime and might be unnecessary since we’re already waiting for the button to be found. Plus, the bot detection happens at the request level, and we already have a delay right after the button is clicked.
@chaodonghu hmm so I guess the following button only triggers the confirmation html div ... right?
@rashadraz we could discuss privately if you need the script modified for this purpose tho ..
@chaodonghu please look into the unfollow interval .. it generates upto a 45 second delay .. i will look to modify the script on my end
& also this is better ..
const INTERVAL_SECONDS = Math.round(UNFOLLOW_INTERVAL / 1000);
console.log(`Wait ${INTERVAL_SECONDS} seconds`);
i was thinking .. what about a code that randomly picks between 5000ms - 10000ms for the interval
UPDATE: here's the code . you can put the numbers in variables ..
const UNFOLLOW_INTERVAL = Math.floor(Math.random() * (10000 - 5000 + 1) + 5000);
minor improvrments to the script which allow it work in Aug 2023, better handles the time between unfollows, and scrolls down to handle the next batch received by AJAX.