├── .github ├── FUNDING.yml └── icon.png ├── .gitignore ├── LICENSE ├── README.md ├── package-lock.json ├── package.json ├── src ├── UImodul.js ├── captchaSolver.js ├── captchaSolverApi.js ├── config.js ├── index.js └── utils.js └── webpack.config.js /.github/FUNDING.yml: -------------------------------------------------------------------------------- 1 | # These are supported funding model platforms 2 | 3 | github: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2] 4 | patreon: # Replace with a single Patreon username 5 | open_collective: # Replace with a single Open Collective username 6 | ko_fi: red_official 7 | tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel 8 | community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry 9 | liberapay: # Replace with a single Liberapay username 10 | issuehunt: # Replace with a single IssueHunt username 11 | otechie: # Replace with a single Otechie username 12 | lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry 13 | custom: #['https://www.paypal.com/donate/?hosted_button_id=LTL43KCDUWVPL'] 14 | -------------------------------------------------------------------------------- /.github/icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/LetsUpdate/CSN/769994b8e7f283559937b0663ebef15e60bce4d9/.github/icon.png -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # Node.js 2 | node_modules/ 3 | npm-debug.log* 4 | yarn-debug.log* 5 | yarn-error.log* 6 | 7 | # Logs 8 | logs/ 9 | *.log 10 | npm-debug.log* 11 | yarn-debug.log* 12 | yarn-error.log* 13 | lerna-debug.log* 14 | 15 | # Dependency directories 16 | node_modules/ 17 | jspm_packages/ 18 | 19 | # Optional npm cache directory 20 | .npm 21 | 22 | # Optional eslint cache 23 | .eslintcache 24 | 25 | # Optional REPL history 26 | .node_repl_history 27 | 28 | # Output of 'npm pack' 29 | *.tgz 30 | 31 | # Coverage directory used by tools like istanbul 32 | coverage/ 33 | 34 | # Webpack files 35 | dist/ 36 | build/ 37 | 38 | # IDE specific files 39 | .vscode/ 40 | .idea/ 41 | 42 | # MacOS specific files 43 | .DS_Store 44 | 45 | # Environment variables 46 | .env 47 | .env.local 48 | .env.development.local 49 | .env.test.local 50 | .env.production.local 51 | 52 | # Tampermonkey specific files 53 | *.user.js 54 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | MIT License 2 | 3 | Copyright (c) 2023 RED 4 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy 6 | of this software and associated documentation files (the "Software"), to deal 7 | in the Software without restriction, including without limitation the rights 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 9 | copies of the Software, and to permit persons to whom the Software is 10 | furnished to do so, subject to the following conditions: 11 | 12 | The above copyright notice and this permission notice shall be included in all 13 | copies or substantial portions of the Software. 14 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE 21 | SOFTWARE. 22 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Captcha Solver for Neptun 2 | 3 | 4 | 5 | A **CSN** célja a Captcha kihívások megoldásának automatizálása a Neptun platformon. 6 | 7 | ## Újdonság: a script fixelve, és teljesen újra lett gondolva! 8 | 9 | 10 | 11 | *Dobj egy csillagot hogy tudjak vele menőzni!* [⭐KaptamEgyCsillagot!!!⭐](https://coub.com/view/1uvg42) 12 | 13 | Telepítési útmutató: 14 | 1. Első lépésként telepítsd a böngésződbe a [Tampermonkey](https://www.tampermonkey.net/) kiegészítőt, amely lehetővé teszi a JavaScript alapú szkriptek futtatását a böngésződben. 15 | 2. A szkript telepítéséhez [kattints ide](https://github.com/LetsUpdate/CSN/releases/latest/download/CSN.user.js). Ezután a szkript automatikusan bekerül a Tampermonkey kiegészítőbe, és már el is kezdheted használni! 16 | 3. **Az első használatkor engedélyt fog kérni a backend szerverre való csatlakozáshoz, engedélyezd ezt neki.** 17 | 18 | **Figyelem:** A script csak a captcha-t küldi el és a szerver is csak azt képes elfogadni. Ha nem hiszed, nézd meg a kódot! 19 | 20 | *tipp: Ha nem működne a PowerUp [@boglarka](https://github.com/boglarkla) írt rá egy fixet: [boglarka/npufix](https://github.com/boglarkla/npu/)* 21 | 22 | 23 | 24 | ## 25 | *Ha nagyon tetszett, akár [meghívhatsz egy kávéra](https://ko-fi.com/red_official)* 26 | 27 | [![ko-fi](https://ko-fi.com/img/githubbutton_sm.svg)](https://ko-fi.com/Q5Q0O1LDA) 28 | 29 | koszi :P 30 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "csn2", 3 | "version": "1.0.3", 4 | "description": "Captcha Solver For Neptun", 5 | "main": "src/index.js", 6 | "scripts": { 7 | "build:prod": "webpack", 8 | "build:dev": "cross-env NODE_ENV=development webpack", 9 | "watch": "cross-env NODE_ENV=development webpack --watch", 10 | "serve": "cross-env NODE_ENV=development webpack serve --mode development" 11 | }, 12 | "author": "RED", 13 | "license": "MIT", 14 | "devDependencies": { 15 | "cross-env": "^7.0.3", 16 | "eslint": "^8.36.0", 17 | "eslint-config-prettier": "^8.7.0", 18 | "npm-run-all": "^4.1.5", 19 | "prettier": "^2.8.4", 20 | "terser-webpack-plugin": "^5.3.10", 21 | "webpack": "^5.93.0", 22 | "webpack-cli": "^5.1.4", 23 | "webpack-dev-server": "^5.0.4", 24 | "webpack-userscript": "^3.2.2" 25 | } 26 | } 27 | -------------------------------------------------------------------------------- /src/UImodul.js: -------------------------------------------------------------------------------- 1 | const icons = { 2 | "logo": "https://raw.githubusercontent.com/LetsUpdate/CSN/main/.github/icon.png", 3 | "networkError": "", 4 | "scriptError": "", 5 | } 6 | 7 | const id = "CSN_LOGO" 8 | 9 | function getCaptchaWindow() { 10 | return $('#step2Captcha > div:nth-child(1)') 11 | } 12 | 13 | 14 | function PlaceLogo() { 15 | const style = document.createElement('style'); 16 | style.type = 'text/css'; 17 | style.innerHTML = ` 18 | @keyframes spin { 19 | from { transform: rotate(0deg); } 20 | to { transform: rotate(360deg); } 21 | } 22 | `; 23 | document.getElementsByTagName('head')[0].appendChild(style); 24 | 25 | getCaptchaWindow().append( 26 | `
27 | logo 28 |
` 29 | ) 30 | //onclick event open https://github.com/LetsUpdate/CSN 31 | $(`#${id}`).click(() => { 32 | window.open('https://github.com/LetsUpdate/CSN', '_blank'); 33 | }); 34 | 35 | } 36 | 37 | const LogoState = { 38 | "default": "default", 39 | "solving": "solving", 40 | "error":"error", 41 | "networkError":"networkError", 42 | } 43 | 44 | function SetState(state){ 45 | const logo = $(`#${id} > img`) 46 | switch(state){ 47 | case LogoState.default: 48 | logo.attr('src', icons.logo) 49 | logo.css('animation', 'none') 50 | break; 51 | case LogoState.solving: 52 | logo.attr('src', icons.logo) 53 | logo.attr('title', 'Megoldás folyamatban...') 54 | //rotate the logo 55 | logo.css({ 56 | 'animation': 'spin 0.8s linear infinite', 57 | '-webkit-animation': 'spin 0.8s linear infinite', // For Safari 58 | '-moz-animation': 'spin 0.8s linear infinite', // For Firefox 59 | '-o-animation': 'spin 0.8s linear infinite' // For Opera 60 | }); 61 | break; 62 | case LogoState.error: 63 | logo.attr('src', icons.scriptError) 64 | logo.css('animation', 'none') 65 | //add hint 66 | logo.attr('title', 'Ismeretlen hiba a captcha megoldása közben') 67 | break; 68 | case LogoState.networkError: 69 | logo.attr('src', icons.networkError) 70 | logo.css('animation', 'none') 71 | logo.attr('title', 'CSN Szerver Hiba / unreachable') 72 | break; 73 | } 74 | } 75 | 76 | 77 | module.exports = { PlaceLogo, SetState, LogoState } -------------------------------------------------------------------------------- /src/captchaSolver.js: -------------------------------------------------------------------------------- 1 | const { solveCaptcha } = require('./captchaSolverApi'); 2 | const { SetState, LogoState } = require('./UImodul'); 3 | 4 | const $ = window.jQuery; 5 | 6 | //refreshIcon 7 | function getCaptchaRefreshIcon() { 8 | return $(".captchaRefreshIcon"); 9 | } 10 | //captchaImage 11 | function getCaptchaImage() { 12 | return $("#upCaptchaLogin_loginCaptcha > img"); 13 | } 14 | // 15 | function getCaptchaInput() { 16 | return $("#txtCaptchaLogin"); 17 | } 18 | // 19 | //body > div.ui-dialog.ui-widget.ui-widget-content.ui-corner-all.ui-front.ui-dialog-buttons.ui-draggable > div.ui-dialog-buttonpane.ui-widget-content.ui-helper-clearfix > div > button:nth-child(2) 20 | function getNewLoginButton() { 21 | // Use a more specific selector to target buttons with text containing 'Belépés' or 'login' (case-insensitive) 22 | return $("button.ui-button").filter(function () { 23 | return /belépés|login/i.test($(this).text()); 24 | }); 25 | } 26 | // 27 | function getFirstLogin() { 28 | return $("#btnSubmit"); 29 | } 30 | 31 | function getErrorLabel() { 32 | return $("#txtCaptchaLogin-error"); 33 | } 34 | 35 | function scanForElementText(element, timeout = 10000) { 36 | return new Promise((resolve, reject) => { 37 | const intervalTime = 100; // Interval time in milliseconds 38 | let elapsedTime = 0; 39 | 40 | const interval = setInterval(() => { 41 | if (element === null) { return; } 42 | if (element.text().trim().length > 0) { 43 | clearInterval(interval); 44 | resolve(false); // Captcha is invalid 45 | } 46 | 47 | elapsedTime += intervalTime; 48 | if (elapsedTime >= timeout) { 49 | clearInterval(interval); 50 | reject("TIME_OUT"); // Captcha is valid (no error message appeared within the timeout) 51 | } 52 | }, intervalTime); 53 | }); 54 | } 55 | let tryes = 0; 56 | async function StartSolving(imgsrc) { 57 | 58 | if (tryes > 4) { 59 | SetState(LogoState.error); 60 | return; 61 | } 62 | SetState(LogoState.solving); 63 | tryes++; 64 | 65 | 66 | const captchaImage = getCaptchaImage(); 67 | const loginButton = getNewLoginButton(); 68 | const captchaInput = getCaptchaInput() 69 | const captchaRefreshIcon = getCaptchaRefreshIcon(); 70 | 71 | try { 72 | console.log('Solving captcha...'); 73 | await new Promise((resolve) => setTimeout(resolve, 100)); 74 | let captchaSolution; 75 | if (imgsrc === captchaImage[0].src) { 76 | captchaSolution = await solveCaptcha(captchaImage[0]); 77 | } else { 78 | console.log('Captcha image changed, terminating...'); 79 | return; 80 | } 81 | 82 | 83 | captchaInput.val(captchaSolution); 84 | 85 | if (imgsrc === captchaImage[0].src) { 86 | await new Promise((resolve) => setTimeout(resolve, 100)); 87 | loginButton.click(); 88 | console.log('Captcha solved:', captchaSolution); 89 | } else { 90 | console.log('Captcha image changed, terminating...'); 91 | return; 92 | } 93 | 94 | // Ne spammeljük a szervert szét 95 | await new Promise((resolve) => setTimeout(resolve, 500)); 96 | 97 | 98 | //Mutotion observer observe the error label 99 | //if the captcha is invalid the error label will not be empty 100 | 101 | 102 | //wait for the error label to appear 103 | console.log('Waiting for error label...'); 104 | try { 105 | await scanForElementText(getErrorLabel()); 106 | console.log('Captcha is invalid:', getErrorLabel().text()); 107 | getErrorLabel().text(""); 108 | console.log('waitforCaptchaRefresh'); 109 | captchaRefreshIcon.click(); 110 | return; 111 | } 112 | catch (error) { 113 | if (error === "TIME_OUT") { 114 | console.log('TimedOut'); 115 | } else { 116 | console.error('Failed to solve captcha:', error); 117 | } 118 | return; 119 | } 120 | 121 | } 122 | catch (error) { 123 | 124 | if (error === "HARD_IMAGE") { 125 | captchaRefreshIcon.click(); 126 | 127 | } else { 128 | 129 | console.error('Failed to solve captcha:', error); 130 | SetState(LogoState.networkError); 131 | return; 132 | } 133 | } 134 | } 135 | 136 | 137 | 138 | 139 | function handleImageLoad() { 140 | console.log('Image loaded:', getCaptchaImage()[0].src); 141 | StartSolving(getCaptchaImage()[0].src); 142 | } 143 | 144 | 145 | 146 | function LookForCaptcha() { 147 | 148 | const $imgElement = getCaptchaImage(); 149 | const imgElement = $imgElement[0]; 150 | 151 | //onbutton click reset tryes 152 | getFirstLogin().click(function () { 153 | tryes = 0; 154 | }); 155 | 156 | // Check if the element exists 157 | if (imgElement) { 158 | // Create a new MutationObserver 159 | const observer = new MutationObserver((mutations) => { 160 | mutations.forEach((mutation) => { 161 | if (mutation.type === 'attributes' && mutation.attributeName === 'src') { 162 | console.log('Image src changed to:', imgElement.src); 163 | 164 | 165 | // Attach a load event listener to ensure the image is loaded 166 | $imgElement.off('load').on('load', function () { 167 | if (this.complete && this.naturalWidth > 0) { 168 | handleImageLoad(); 169 | } 170 | }); 171 | 172 | // Also handle the case where the image might already be cached 173 | // if (imgElement.complete && imgElement.naturalWidth > 0) { 174 | // console.log('Image loaded from cache:', imgElement.src); 175 | // handleImageLoad(); 176 | // } 177 | 178 | 179 | } 180 | }); 181 | }); 182 | 183 | // Start observing the image element for changes in its attributes 184 | observer.observe(imgElement, { 185 | attributes: true // Listen for attribute changes 186 | }); 187 | } else { 188 | console.error('Image element not found'); 189 | } 190 | 191 | console.log('Observing document body for changes...'); 192 | } 193 | 194 | 195 | 196 | module.exports = { LookForCaptcha }; -------------------------------------------------------------------------------- /src/captchaSolverApi.js: -------------------------------------------------------------------------------- 1 | const config = require('./config'); 2 | 3 | function convertImageToBase64(imageElement) { 4 | // Create a canvas element 5 | const canvas = document.createElement('canvas'); 6 | const context = canvas.getContext('2d'); 7 | 8 | // Set canvas dimensions to match the image 9 | canvas.width = imageElement.naturalWidth; 10 | canvas.height = imageElement.naturalHeight; 11 | 12 | // Draw the image onto the canvas 13 | context.drawImage(imageElement, 0, 0); 14 | 15 | // Convert the canvas content to a Data URL (Base64 string) 16 | const base64String = canvas.toDataURL('image/jpeg'); 17 | 18 | // Resolve the promise with the Base64 string 19 | return base64String; 20 | 21 | } 22 | 23 | // if return message is "HARD_IMAGE" then the image is too hard to solve rigt now 24 | function solveCaptcha(imageElement) { 25 | return new Promise((resolve, reject) => { 26 | // Convert the canvas content to a Data URL (Base64 string) 27 | const base64String = convertImageToBase64(imageElement); 28 | 29 | if(base64String.length < 100) { 30 | reject('BASE64_CONVERSION_FAILED'); 31 | return; 32 | } 33 | 34 | // Make the GM_xmlhttpRequest with the Base64 string 35 | GM_xmlhttpRequest({ 36 | method: 'POST', 37 | url: config.API_URL, // Replace with your API endpoint 38 | headers: { 39 | 'Content-Type': 'application/json' 40 | }, 41 | data: JSON.stringify({ image64: base64String }), 42 | onload: function(response) { 43 | if (response.status === 200) { 44 | try { 45 | const jsonResponse = JSON.parse(response.responseText); 46 | if(jsonResponse.message === "HARD_IMAGE") { 47 | reject(jsonResponse.message); 48 | 49 | }else{ resolve(jsonResponse.message);} 50 | 51 | 52 | } catch (error) { 53 | reject(new Error('Failed to parse response JSON')); 54 | } 55 | } else { 56 | console.error('Failed to solve captcha:', JSON.parse(response.responseText).error); 57 | reject(new Error('Failed to solve captcha')); 58 | } 59 | }, 60 | onerror: function(error) { 61 | reject(error); 62 | } 63 | }); 64 | }); 65 | } 66 | 67 | 68 | 69 | // Export the function for use in other modules 70 | module.exports = { solveCaptcha, convertImageToBase64 }; -------------------------------------------------------------------------------- /src/config.js: -------------------------------------------------------------------------------- 1 | const config = { 2 | API_URL: 'https://csn.redsparks.eu/api/v1/solve', 3 | } 4 | //config.API_URL = 'http://localhost:8000/api/v1/solve'; 5 | module.exports = config; -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | const utils = require("./utils"); 2 | const { LookForCaptcha } = require('./captchaSolver'); 3 | const {PlaceLogo} = require('./UImodul'); 4 | 5 | 6 | 7 | (function () { 8 | 'use strict'; 9 | 10 | if (utils.isNeptunPage() && utils.isLoginPage()) { 11 | // Set up a Mutation Observer to watch for changes in the DOM 12 | 13 | //Prevent npu to press login after the user too 14 | document.addEventListener('keydown', function(event) { 15 | if (event.key === 'Enter') { 16 | const $abortLoginLink = $('#abortLogin > a'); 17 | if ($abortLoginLink.length) { 18 | $abortLoginLink[0].click(); 19 | } 20 | } 21 | }); 22 | console.log("LookForCaptcha"); 23 | PlaceLogo(); 24 | LookForCaptcha(); 25 | } 26 | })(); -------------------------------------------------------------------------------- /src/utils.js: -------------------------------------------------------------------------------- 1 | const $ = window.jQuery; 2 | 3 | // Verify that we are indeed on a Neptun page 4 | function isNeptunPage() { 5 | return document.title.toLowerCase().indexOf("neptun.net") !== -1; 6 | } 7 | 8 | // Returns whether we are on the login page 9 | function isLoginPage() { 10 | return $("td.login_left_side").size() > 0; 11 | } 12 | 13 | module.exports = { 14 | isNeptunPage, 15 | isLoginPage, 16 | }; 17 | -------------------------------------------------------------------------------- /webpack.config.js: -------------------------------------------------------------------------------- 1 | const path = require('path'); 2 | const { UserscriptPlugin } = require('webpack-userscript'); 3 | const TerserPlugin = require('terser-webpack-plugin'); 4 | 5 | 6 | const dev = process.env.NODE_ENV === 'development'; 7 | 8 | module.exports = { 9 | mode: dev ? 'development' : 'production', 10 | entry: path.resolve(__dirname, 'src', 'index.js'), // Your entry point 11 | output: { 12 | path: path.resolve(__dirname, 'dist'), 13 | filename: 'CSN.user.js', // Output file 14 | }, 15 | devServer: { 16 | static: { 17 | directory: path.join(__dirname, 'dist'), 18 | }, 19 | client: false, // Disable client injection 20 | hot: false, // Disable hot reloading 21 | 22 | 23 | }, 24 | optimization: { 25 | minimize: dev ? false : true, 26 | minimizer: [ 27 | new TerserPlugin({ 28 | terserOptions: { 29 | compress: { 30 | drop_console: dev ? false : true, 31 | }, 32 | }, 33 | }), 34 | ], 35 | }, 36 | plugins: [ 37 | new UserscriptPlugin({ 38 | headers(original) { 39 | const baseHeaders = { 40 | name: 'Captcha Solver For Neptun', 41 | namespace: 'https://github.com/LetsUpdate/CSN', 42 | description: 'No captcha 4 u', 43 | author: 'RED', 44 | include: [ 45 | 'https://*neptun*/*hallgato*/*', 46 | 'https://*neptun*/*Hallgatoi*/*', 47 | 'https://*hallgato*.*neptun*/*', 48 | 'https://netw*.nnet.sze.hu/hallgato', 49 | 'https://nappw.dfad.duf.hu/hallgato/*', 50 | ], 51 | grant: 'GM_xmlhttpRequest', 52 | "run-at": "document-end", 53 | require: "https://ajax.googleapis.com/ajax/libs/jquery/1.11.1/jquery.min.js", 54 | updateURL: "https://github.com/LetsUpdate/CSN/releases/latest/download/CSN.meta.js", 55 | downloadURL:"https://github.com/LetsUpdate/CSN/releases/latest/download/CSN.user.js", 56 | supportURL:"https://github.com/LetsUpdate/CSN", 57 | icon64: "https://raw.githubusercontent.com/LetsUpdate/CSN/main/.github/icon.png", 58 | license: "MIT", 59 | 60 | }; 61 | 62 | if (dev) { 63 | return { 64 | ...original, 65 | ...baseHeaders, 66 | version: `${original.version}-build.[buildNo]`, 67 | }; 68 | } 69 | 70 | return { 71 | ...original, 72 | ...baseHeaders, 73 | }; 74 | }, 75 | 76 | }), 77 | 78 | ], 79 | }; --------------------------------------------------------------------------------