Instagram Chat Bot

A script that adds a side panel to Instagram's DM section. Activates an AI-powered bot to automatically reply to incoming messages.

// ==UserScript==
// @name         Instagram Chat Bot
// @namespace    https://github.com/swempish
// @version      1.0.6
// @description  A script that adds a side panel to Instagram's DM section. Activates an AI-powered bot to automatically reply to incoming messages.
// @author       Emirhan Çolak
// @match        https://www.instagram.com/direct/t/*
// @icon         
// @grant        none
// @license GPL-3.0
// @require      https://code.jquery.com/jquery-3.7.1.slim.min.js
// ==/UserScript==

// Side panel element (Start bot button)
var panelElement = `

<div style="
    justify-content: center;
    display: flex;
    flex-direction: column;
    text-align: center;
    padding: 1rem;
    gap: 1rem;
">
    <h2>Insta Chat Bot Utilz</h2>
    <button id="startTheBot" style="
    min-height: 2rem;
    max-width: 14rem;
    margin: auto;
    width: 100%;
    cursor: pointer;
">Start the Bot!</button>
    <div style="
    border-style: outset;
    border-color: brown;
    background-color: rgb(66,66,66);
    text-align: left;
    padding: 0 .2rem 0 .2rem;
    font-family: monospace;
    font-size: x-small;
    word-break: break-all;
    overflow-x: hidden;
    overflow-y: auto;
    max-height: 8rem;
    height: 100%;
    display: none;
" id="botLogChat">
        <p>Log</p>
    </div>
</div>

`;

const injectCSS = css => {
    let el = document.createElement('style');
    el.type = 'text/css';
    el.innerText = css;
    document.head.appendChild(el);
    return el;
};

var days = ["Sunday", "Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday"];

var personalities = [
    'Your name is Emirhan. You are talking to someone in Instagram. Reply to them.' // Put here your personality prompt
]

const GEMINI_API_KEY = "YOUR-GEMINI-API-KEY";
const GEMINI_API_URL = `https://generativelanguage.googleapis.com/v1beta/models/gemini-pro:generateContent?key=${GEMINI_API_KEY}`;

var bot_config = {
    contents: [
        {
            role: "user",
            parts: [{ text: personalities[0] }]
        },
        {
            role: "model",
            parts: [{ text: "Okay." }]
        },
    ],
    "safetySettings": [
        {
            category: "HARM_CATEGORY_HARASSMENT",
            threshold: "BLOCK_NONE"
        },
        {
            category: "HARM_CATEGORY_HATE_SPEECH",
            threshold: "BLOCK_NONE"
        },
        {
            category: "HARM_CATEGORY_SEXUALLY_EXPLICIT",
            threshold: "BLOCK_NONE"
        },
        {
            category: "HARM_CATEGORY_DANGEROUS_CONTENT",
            threshold: "BLOCK_NONE"
        },
    ]
}

async function generate_ai_response(input) {

    // Provide time information to the bot
    //save_message("*** SYSTEM MESSAGE *** Time: " + new Date().toLocaleTimeString(undefined, { hour: '2-digit', minute: '2-digit' }) + " Date: " + new Date().toLocaleDateString() + " Day of the week: " + days[new Date().getDay()], "user");
//save_message("*** DATE AND TIME RECORDED ***", "model");

    save_message(input, "user");
    const response = await fetch(GEMINI_API_URL, {
        method: "POST",
        body: JSON.stringify(bot_config),
    });

    const result = await response.json();
    save_message(result["candidates"][0]["content"]["parts"][0]["text"], "model");
    return String(result["candidates"][0]["content"]["parts"][0]["text"]);
}

function save_message(message, role) {
    bot_config["contents"].push({ role: role, parts: [{ text: message }] });
}

var lastSentMessage = "---------------";
var generatingResponse = false;

var lastChatContent = "";
var isButtonAvailable = false;

function main() {
    // Add the start bot button
    if(!isButtonAvailable){
        document.querySelector("main > section > div > div > div > div > div > div > div > div:nth-child(2) > div > div > div > div > div:nth-child(2)").innerHTML = panelElement + document.querySelector("main > section > div > div > div > div > div > div > div > div:nth-child(2) > div > div > div > div > div:nth-child(2)").innerHTML;
    isButtonAvailable = true;
    }

    var chatContainer = document.querySelector("main > section > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div:nth-child(3) > div");
    var sendButton = document.querySelector("main > section > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div:nth-child(2) > div > div > div:nth-child(3)");

    document.querySelector("#startTheBot").addEventListener("click", function () {
        var inputPlaceholderElement = document.querySelector("main > section > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div > div:nth-child(2) > div > div > div > div > div:nth-child(2)");
        inputPlaceholderElement.innerText = "Controlled by the bot...";
        $("#botLogChat").show();
        logBot("Starting the bot...");
        injectCSS('*{--messenger-card-background: rgb(34 113 120);}');
        lastChatContent = chatContainer.innerHTML;

        // Create a new interval that will check chatContainer's innerHTML and if it changes, generate a response
        var elementTesterInterval = setInterval(async () => {
            if (!generatingResponse && chatContainer.innerHTML != lastChatContent) {
                var chatSize = chatContainer.childNodes.length;
                var lastMessage = "";

                for (let index = 1; index < chatContainer.childNodes[chatSize - 2].getElementsByTagName("span").length; index++) {
                    lastMessage += chatContainer.childNodes[chatSize - 2].getElementsByTagName("span")[index].innerText + " ";
                }

                var whoSent = chatContainer.childNodes[chatSize - 2].getElementsByTagName("span")[0].innerText == "You sent" ? "yourself" : "other";

                if (generatingResponse || whoSent == "yourself") {
                    return;
                }

                logBot("Got message from user. Waiting for AI response...");
                generatingResponse = true;
                var response = await generate_ai_response(lastMessage);

                logBot(`Got the response: ${response.slice(0, 30) + "..."}`);

                lastSentMessage = response;
                typeAndSend(response, sendButton);
            }
        }, 100);

        var observer = new MutationObserver(function (mutations) {
            // Iterate over each mutation record
            mutations.forEach(async function (mutation) {
                if (mutation.type === 'childList') {

                }
            });
        });

        // Configure the observer to observe changes in childList
        var config = { childList: true };

        // Start observing the chat container element
        observer.observe(chatContainer, config);


    });
}

async function typeAndSend(text, sendButton) {
    // Insert text character by character so it looks like someone is typing. Max duration is 1 second
    // This is to simulate human typing
    for (var i = 0; i < text.length; i++) {
        document.execCommand('insertText', false, text.charAt(i));
        await new Promise(r => setTimeout(r, 100));
    }
    // Wait for 1 second
    await new Promise(r => setTimeout(r, 100));
    sendButton.click();
    generatingResponse = false;
    await new Promise(r => setTimeout(r, 100));
    inputPlaceholderElement.innerText = "Controlled by the bot...";
}

// Log to chat div. Put logs to bottom
function logBot(text) {
    document.querySelector("#botLogChat").innerHTML += "<p>" + text + "</p>";
    document.querySelector("#botLogChat").scrollTop = document.querySelector("#botLogChat").scrollHeight;
}


$(document).ready(function () {
    var elementTesterInterval = setInterval(() => {
        if (document.querySelector("main > section > div > div > div > div > div > div > div > div:nth-child(2) > div > div > div > div > div:nth-child(2)")) {
            clearInterval(elementTesterInterval);
            main();
        }
    }, 1000);
});