Naurok Bypass v2

Fetches answers to *all* Naurok quizes

目前为 2023-03-12 提交的版本。查看 最新版本

// ==UserScript==
// @name        Naurok Bypass v2
// @namespace   https://gf.qytechs.cn/users/1028724-griffi-gh
// @match       https://naurok.com.ua/test/*.html
// @grant       none
// @version     1.6
// @author      griffi-gh
// @run-at      document-end
// @inject-into page
// @description Fetches answers to *all* Naurok quizes
// @license     MIT
// @icon        
// ==/UserScript==
"use strict";

const LOADING_IMAGE = "https://media.tenor.com/MlCeUwzn2nEAAAAM/troll-lol.gif";
const TEST_IMAGE_URL = "https://cdn-icons-png.flaticon.com/512/190/190411.png";
const TEST_UPDATE_IMAGE_URL = "https://cdn-icons-png.flaticon.com/512/1601/1601884.png";

const ls_key = `cached-kettle-smoke-${document.location.pathname.replaceAll("/", "-").slice(1, -5).toLowerCase()}`;

async function loadStuff() {
  pre_display()

  //Get test path
  const base = document.location.href.slice(0, -5);

  //Load the homework creation page
  const set_text = await fetch(base + "/set").then(x => x.text());
  const set_document = document.createElement("html");
  set_document.innerHTML = set_text;

  //Get form data and modify it
  const set_form = set_document.querySelector("#w0");
  const set_form_data = new FormData(set_form);
  set_form_data.set("Homework[deadline_day]", "9999-01-01");
  set_form_data.set("Homework[show_answer]", "1");

  //Start homework
  const homework_res = await fetch(set_form.action, {
    method: 'POST',
    credentials: 'include',
    body: set_form_data,
  });

  //Get homework url and id
  const homework_url = homework_res.url;
  const homework_id = homework_url.split("/").at(-1);
  console.log("Homework url:", homework_url);

  //Get homework document object
  const homework_text = await homework_res.text();
  const homework_document = document.createElement("html");
  homework_document.innerHTML = homework_text;

  //Get the CSRF token and create FormData from it
  const homework_csrf_param = homework_document.querySelector('meta[name="csrf-param"]').content
  const homework_csrf_token = homework_document.querySelector('meta[name="csrf-token"]').content
  const homework_csrf_form_data = new FormData();
  homework_csrf_form_data.set(homework_csrf_param, homework_csrf_token);

  //Get join code
  const homework_code = homework_document.querySelector(".homework-code").textContent;

  //Load the join page
  const join_text = await fetch(`/test/join?gamecode=${homework_code}`).then(res => res.text());
  const join_document = document.createElement('html');
  join_document.innerHTML = join_text;

  //Get form data and fill in the username
  const join_form = join_document.querySelector("#participate-form-code");
  const join_form_data = new FormData(join_form);
  join_form_data.set("JoinForm[name]", "[object Object]"); //troll naurok developers while we're at it lol

  //Start homework session
  const join_res = await fetch(join_form.action, {
    method: 'POST',
    credentials: 'include',
    body: new URLSearchParams(join_form_data).toString(),
    headers: {
      "Content-Type": "application/x-www-form-urlencoded",
    },
  });

  //Get test session address
  const test_session_url = join_res.url;

  //Get the test session document
  const session_text = await fetch(test_session_url).then(res => res.text());
  const session_document = document.createElement("html");
  session_document.innerHTML = session_text;

  //Get session id
  const testik_elem = session_document.querySelector('[ng-app="testik"]');
  const ng_init = testik_elem.getAttribute("ng-init");
  const ng_init_numbers = ng_init.match(/[0-9]+/g);
  const session_id = ng_init_numbers[1] || 0;
  console.log("Session id", session_id);

  //Get session info
  const session_info = await fetch(`https://naurok.com.ua/api2/test/sessions/${session_id}/`, {
    "credentials": "omit",
  }).then(x => x.json());
  console.log("Session info: ", session_info);

  //Find the latest question
  const {latest_question, questions} = session_info;
  const question = latest_question ? questions.find(question => question.id == latest_question) : questions[0];

  //Get the answer id
  const answer_id = question.options[0].id.toString();

  //Answer the question
  await fetch("/api2/test/responses/answer", {
    "method": "PUT",
    "credentials": "include",
    headers: {
      'Accept': 'application/json, text/plain, */*',
      'Content-Type': 'application/json'
    },
    "body": JSON.stringify({
      "session_id": session_id,
      "answer": [answer_id],
      "question_id": question.id,
      "show_answer": 1,
      "type": "quiz",
      "point": question.point.toString(),
      "homeworkType": question.type,
      "homework": true
    }),
  });

  //End the session
  const end_sess_data = await fetch(`/api2/test/sessions/end/${session_id}`, {
    "method": "PUT",
    "credentials": "include",
  }).then(res => res.json());
  const end_sess_uuid = end_sess_data.session.uuid;

  //Fetch the end page
  const test_end_text = await fetch(`/test/complete/${end_sess_uuid}`, {
    redirect: 'follow'
  }).then(res => res.text());

  //Get the text end document
  const test_end_document = document.createElement("html");
  test_end_document.innerHTML = test_end_text;

  //Get the answers
  const answers = test_end_document.querySelector(".homework-stats");

  //Display answers
  display_answers(answers);

  //Stop homework
  await fetch(`/test/homework/${homework_id}/stop`, {
    method: 'POST',
    credentials: 'include',
    body: homework_csrf_form_data,
  });

  //Delete homework
  await fetch(`/test/homework/${homework_id}/delete`, {
    method: 'POST',
    credentials: 'include',
    body: homework_csrf_form_data,
  });

  return answers
};

function pre_display() {
  //Delete previously displayed
  Array.from(document.querySelectorAll(".question-view-item")).forEach(item => item.remove());

  //Clear the regular questions
  Array.from(document.querySelectorAll(".chicken-beef")).forEach(item => item.remove());
}

function display_answers(answers) {
  pre_display()

  //Add classes
  answers.classList.add("row");
  answers.classList.add("chicken-beef")

  //HACK: Remove "- your answer" text
  answers.innerHTML = answers.innerHTML.replaceAll("<em>— ваша відповідь</em>", "");

  //Add answers to the page
  const afer_element = document.querySelector(".block-head");
  afer_element.parentNode.insertBefore(answers, afer_element.nextSibling);
};

async function loadStuffWriteCache() {
  const answers = await loadStuff();
  const elem = document.createElement("div");
  display_answers(elem);
  elem.appendChild(answers.cloneNode(true));
  localStorage.setItem(ls_key, elem.innerHTML);
}

//Preload images
{
  const temp_image = new Image();
  temp_image.src = LOADING_IMAGE;
  window._cow_taxes = temp_image;
}

//Add CSS
{
  const style = `
    .chicken-beef {
      padding: 1.33rem;
    }
    .chicken-beef .incorect {
      background-color: #d8d8d8 !important;
    }
    .chicken-beef .content-block .question-label,
    .chicken-beef .content-block.success,
    .chicken-beef .content-block.skipped,
    .chicken-beef .content-block.failed {
      display: none;
    }
  `;
  const style_elem = document.createElement("style");
  style_elem.textContent = style;
  document.head.appendChild(style_elem);
}

//Load cached answers
let is_cached = false;
if (localStorage.getItem(ls_key)) {
  const elem = document.createElement("div");
  elem.innerHTML = localStorage.getItem(ls_key);
  display_answers(elem.firstChild);
  is_cached = true;
}

//Create answers button
{
  const button = document.createElement("button");
  button.type = "button";
  button.classList.add("test-action-button");
  button.innerHTML = `
    <img src="${TEST_IMAGE_URL}">
    <span>Завантажити ідповіді</span>
  `;
  button.style = `
    display: flex;
    width: 100%;
    border-width: 0;
    font-family: inherit;
    font-size: inherit;
    font-style: inherit;
    font-weight: inherit;
    line-height: inherit;
  `;
  if (is_cached) {
    button.querySelector('img').src = TEST_UPDATE_IMAGE_URL;
    button.querySelector('span').textContent = "Оновити відповіді";
  }
  button.addEventListener("click", async () => {
    button.querySelector('img').src = LOADING_IMAGE;
    button.querySelector('span').textContent = "Завантаження...";
    await loadStuffWriteCache();
    button.querySelector('img').src = TEST_UPDATE_IMAGE_URL;
    button.querySelector('span').textContent = "Оновити відповіді";
  });
  const buttons = document.querySelector(".single-test-actions");
  buttons.prepend(button);
}

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址