Onlyfans "leak" finder

Checks some sites for OF "leaks"

  1. // ==UserScript==
  2. // @name Onlyfans "leak" finder
  3. // @namespace http://tampermonkey.net/
  4. // @version 0.8.1.1
  5. // @description Checks some sites for OF "leaks"
  6. // @author You
  7. // @match https://onlyfans.com/*
  8. // @match *://*.coomer.su/*
  9. // @match *://*.fapello.com/*
  10. // @match https://fansly.com/*
  11. // @match https://fantrie.com/*
  12. // @match *://*.topfapgirls1.com/*
  13. // @grant GM.xmlHttpRequest
  14. // @license Unlicense
  15. // ==/UserScript==
  16.  
  17. (function () {
  18. 'use strict';
  19.  
  20. const MAX_RETRIES = 5;
  21.  
  22. function getUsernameFromUrl() {
  23. const hostname = window.location.hostname;
  24. const pathname = window.location.pathname;
  25.  
  26. if (hostname === 'onlyfans.com' || hostname === 'fantrie.com') {
  27. return pathname.split('/')[1];
  28. }
  29.  
  30. if (hostname === 'fansly.com') {
  31. const matches = pathname.match(/^\/([^/]+)/);
  32. return matches ? matches[1] : '';
  33. }
  34.  
  35. return '';
  36. }
  37.  
  38. function createRequest(url, onSuccess, onError, retryCount = 0) {
  39. GM.xmlHttpRequest({
  40. method: 'GET',
  41. url: url,
  42. onload(response) {
  43. if (response.status === 200) {
  44. onSuccess(response);
  45. } else if (retryCount < MAX_RETRIES) {
  46. console.log(`Retrying... Attempt ${retryCount + 1}`);
  47. setTimeout(() => createRequest(url, onSuccess, onError, retryCount + 1), 1000);
  48. } else {
  49. console.error('Max retry attempts reached');
  50. onError(response);
  51. }
  52. },
  53. onerror: onError,
  54. });
  55. }
  56.  
  57. function fetchUserProfileFansly(username) {
  58. const apiUrl = `https://apiv3.fansly.com/api/v1/account?usernames=${username}`;
  59.  
  60. createRequest(
  61. apiUrl,
  62. (response) => {
  63. const data = JSON.parse(response.responseText);
  64. if (data.success && data.response.length > 0) {
  65. const id = data.response[0].id;
  66. fetchUserProfileCoomer(id, 'fansly');
  67. } else {
  68. console.log(`User ${username} not found`);
  69. }
  70. },
  71. (error) => console.error('Error fetching Fansly profile:', error)
  72. );
  73. }
  74.  
  75. function fetchUserProfileCoomer(username, service) {
  76. const apiUrl = `https://coomer.su/api/v1/${service}/user/${username}/profile`;
  77.  
  78. createRequest(
  79. apiUrl,
  80. () => {
  81. const profileUrl = `https://coomer.su/${service}/user/${username}`;
  82. addLinkToMenu(profileUrl, 'Coomer.su');
  83. },
  84. (error) => console.error('Error fetching Coomer profile:', error)
  85. );
  86. }
  87.  
  88. function fetchUserProfileFapello(username) {
  89. const apiUrl = `https://fapello.com/${username}/`;
  90.  
  91. createRequest(
  92. apiUrl,
  93. (response) => {
  94. if (response.finalUrl === apiUrl) {
  95. addLinkToMenu(apiUrl, 'Fapello');
  96. } else {
  97. console.log('Request was redirected');
  98. }
  99. },
  100. (error) => console.error('Error fetching Fapello profile:', error)
  101. );
  102. }
  103.  
  104. function fetchUserProfileLeakNudes(username) {
  105. const apiUrl = `https://leaknudes.com/model/${username}`;
  106.  
  107. createRequest(
  108. apiUrl,
  109. (response) => {
  110. if (response.finalUrl === apiUrl) {
  111. addLinkToMenu(apiUrl, 'LeakNudes');
  112. } else {
  113. console.log('Request was redirected');
  114. }
  115. },
  116. (error) => console.error('Error fetching LeakNudes profile:', error)
  117. );
  118. }
  119.  
  120. function fetchUserProfileTopFapGirls(username) {
  121. const searchUrl = `https://www.topfapgirls1.com/search/?q=${username}`;
  122.  
  123. createRequest(
  124. searchUrl,
  125. (response) => {
  126. console.log(response.status)
  127. console.log(searchUrl)
  128. if (response.status === 200 && !response.finalUrl.includes("?q=")) {
  129. console.log(response.finalUrl)
  130. addLinkToMenu(response.finalUrl, 'TopFapGirls1');
  131. } else {
  132. console.log('User not found on TopFapGirls1');
  133. }
  134. },
  135. (error) => console.error('Error fetching TopFapGirls1 profile:', error)
  136. );
  137. }
  138.  
  139. function addLinkToMenu(link, displayText) {
  140. const hostname = window.location.hostname;
  141.  
  142. const createAnchorElement = (link, text) => {
  143. const aElement = document.createElement('a');
  144. aElement.href = link;
  145. aElement.textContent = text;
  146. aElement.style.cssText = `
  147. display: inline-block;
  148. padding: 8px 12px;
  149. margin: 4px 0;
  150. height: 1.8em;
  151. margin: 3.5em 0.25em 0 0.25em;
  152. background-color: #007bff;
  153. color: white;
  154. text-align: center;
  155. border-radius: 4px;
  156. text-decoration: none;
  157. font-size: 14px;
  158. `;
  159. return aElement;
  160. };
  161.  
  162. if (hostname === 'fansly.com') {
  163. const followButton = document.querySelector('.follow-profile');
  164. if (followButton) {
  165. const newMenuElement = followButton.cloneNode(false);
  166. newMenuElement.textContent = displayText;
  167.  
  168. const aElement = createAnchorElement(link, '');
  169. aElement.appendChild(newMenuElement);
  170.  
  171. followButton.after(aElement);
  172. } else {
  173. console.error('Follow button not found');
  174. }
  175. } else if (hostname === 'fantrie.com') {
  176. const menuElement = document.querySelectorAll('.left-menus')[1];
  177. if (menuElement) {
  178. menuElement.appendChild(createAnchorElement(link, displayText));
  179. } else {
  180. console.error('Left menus element not found');
  181. }
  182. } else {
  183. const menuElement = document.querySelector('.l-header__menu.m-native-custom-scrollbar.m-scrollbar-y.m-invisible-scrollbar');
  184. if (menuElement) {
  185. const aElement = createAnchorElement(link, displayText);
  186. aElement.style.marginRight = '10px';
  187.  
  188. if (menuElement.querySelectorAll('a').length > 0) {
  189. menuElement.appendChild(document.createElement('br'));
  190. }
  191.  
  192. menuElement.appendChild(aElement);
  193. } else {
  194. console.error('Menu element not found');
  195. }
  196. }
  197. }
  198.  
  199. function waitForElement(selector, callback) {
  200. const observer = new MutationObserver(() => {
  201. const element = document.querySelector(selector);
  202. if (element) {
  203. observer.disconnect();
  204. callback(element);
  205. }
  206. });
  207.  
  208. observer.observe(document.body, { childList: true, subtree: true });
  209. }
  210.  
  211. const username = getUsernameFromUrl();
  212.  
  213. if (username) {
  214. const hostname = window.location.hostname;
  215. if (hostname === 'fansly.com') {
  216. waitForElement('.profile-details', () => {
  217. fetchUserProfileFansly(username);
  218. fetchUserProfileFapello(username);
  219. fetchUserProfileLeakNudes(username);
  220. fetchUserProfileTopFapGirls(username);
  221. if (username.includes('_')) {
  222. fetchUserProfileFapello(username.replace(/_/g, '-'));
  223. }
  224. const instaElement = document.querySelector(".icon-container.instagram").parentNode
  225. if(instaElement){
  226. const instaHandle = instaElement.href.match(/https:\/\/instagram\.com\/(.+)/)[1];
  227. fetchUserProfileTopFapGirls(instaHandle);
  228. fetchUserProfileFapello(instaHandle);
  229. }
  230. });
  231. } else if (hostname === 'fantrie.com') {
  232. waitForElement('.left-menus', () => {
  233. fetchUserProfileFapello(username);
  234. fetchUserProfileLeakNudes(username);
  235. fetchUserProfileTopFapGirls(username);
  236. });
  237. } else {
  238. waitForElement('.l-header__menu.m-native-custom-scrollbar.m-scrollbar-y.m-invisible-scrollbar', () => {
  239. fetchUserProfileCoomer(username, 'onlyfans');
  240. fetchUserProfileFapello(username);
  241. fetchUserProfileLeakNudes(username);
  242. fetchUserProfileTopFapGirls(username);
  243. if (username.includes('_')) {
  244. fetchUserProfileFapello(username.replace(/_/g, '-'));
  245. }
  246. });
  247. }
  248. }
  249. })();