no comments

This commit is contained in:
Eduard Kuzmenko 2021-02-12 19:24:20 +04:00
parent 04442dcffa
commit e0999ad4d0
19 changed files with 316 additions and 111 deletions

View File

@ -19,64 +19,64 @@
"license": "ISC",
"dependencies": {
"jsbn": "^1.1.0",
"webpack-dev-server": "^3.11.0"
"webpack-dev-server": "^3.11.2"
},
"devDependencies": {
"@babel/cli": "^7.12.1",
"@babel/core": "^7.12.3",
"@babel/plugin-proposal-class-properties": "^7.12.1",
"@babel/plugin-transform-typescript": "^7.12.1",
"@babel/cli": "^7.12.13",
"@babel/core": "^7.12.13",
"@babel/plugin-proposal-class-properties": "^7.12.13",
"@babel/plugin-transform-typescript": "^7.12.13",
"@babel/polyfill": "^7.12.1",
"@babel/preset-env": "^7.12.1",
"@babel/preset-env": "^7.12.13",
"@babel/preset-es2015": "^7.0.0-beta.53",
"@babel/preset-typescript": "^7.12.1",
"@babel/preset-typescript": "^7.12.13",
"@cryptography/aes": "^0.1.1",
"@cryptography/sha1": "^0.1.0",
"@cryptography/sha256": "^0.2.0",
"@peculiar/webcrypto": "^1.1.3",
"@peculiar/webcrypto": "^1.1.6",
"@types/aes-js": "^3.1.1",
"@types/chrome": "0.0.91",
"@types/jest": "^24.9.1",
"@types/serviceworker-webpack-plugin": "^1.0.1",
"aes-js": "^3.1.2",
"autoprefixer": "^9.8.0",
"autoprefixer": "^9.8.6",
"babel-jest": "^24.9.0",
"babel-loader": "^8.1.0",
"babel-loader": "^8.2.2",
"babel-preset-es2015": "^6.24.1",
"compression": "^1.7.4",
"compression-webpack-plugin": "^3.1.0",
"core-js": "^3.6.5",
"css-loader": "^3.5.3",
"core-js": "^3.8.3",
"css-loader": "^3.6.0",
"express": "^4.17.1",
"fast-png": "^5.0.2",
"fast-png": "^5.0.3",
"handlebars": "^4.7.6",
"handlebars-loader": "^1.7.1",
"html-webpack-plugin": "^3.2.0",
"ifdef-loader": "^2.1.5",
"install": "^0.13.0",
"jest": "^24.9.0",
"media-query-plugin": "^1.3.1",
"media-query-plugin": "^1.4.0",
"mini-css-extract-plugin": "^0.9.0",
"node-sass": "^4.14.1",
"npm": "^6.14.10",
"npm": "^6.14.11",
"on-build-webpack": "^0.1.0",
"optimize-css-assets-webpack-plugin": "^5.0.3",
"optimize-css-assets-webpack-plugin": "^5.0.4",
"pako": "^1.0.11",
"postcss": "^7.0.32",
"postcss": "^7.0.35",
"postcss-loader": "^3.0.0",
"postcss-preset-env": "^6.7.0",
"qr-code-styling": "^1.0.1",
"qr-code-styling": "^1.3.4",
"resolve-url-loader": "^3.1.2",
"sass-loader": "^8.0.2",
"serviceworker-webpack-plugin": "^1.0.1",
"style-loader": "^1.2.1",
"terser-webpack-plugin": "^3.0.2",
"style-loader": "^1.3.0",
"terser-webpack-plugin": "^3.1.0",
"ts-jest": "^24.3.0",
"ts-loader": "^6.2.2",
"typescript": "^3.9.7",
"url-loader": "^2.3.0",
"webpack": "^4.43.0",
"webpack-cli": "^3.3.11",
"webpack": "^4.46.0",
"webpack-cli": "^3.3.12",
"webpack-merge": "^4.2.2",
"worker-loader": "^2.0.0"
}

View File

@ -1,22 +1,23 @@
import rootScope from "../../lib/rootScope";
import { generatePathData } from "../../helpers/dom";
//import { generatePathData } from "../../helpers/dom";
import { MyMessage } from "../../lib/appManagers/appMessagesManager";
import Chat from "./chat";
type Group = {bubble: HTMLDivElement, mid: number, timestamp: number}[];
type Group = {bubble: HTMLElement, mid: number, timestamp: number}[];
type BubbleGroup = {timestamp: number, fromId: number, mid: number, group: Group};
export default class BubbleGroups {
private bubbles: Array<BubbleGroup> = []; // map to group
private detailsMap: Map<HTMLElement, BubbleGroup> = new Map();
private groups: Array<Group> = [];
//updateRAFs: Map<HTMLDivElement[], number> = new Map();
//updateRAFs: Map<HTMLElement[], number> = new Map();
private newGroupDiff = 121; // * 121 in scheduled messages
constructor(private chat: Chat) {
}
removeBubble(bubble: HTMLDivElement, mid: number) {
const details = this.bubbles.findAndSplice(g => g.mid === mid);
removeBubble(bubble: HTMLElement) {
const details = this.detailsMap.get(bubble);
if(details && details.group.length) {
details.group.findAndSplice(d => d.bubble === bubble);
if(!details.group.length) {
@ -27,7 +28,7 @@ export default class BubbleGroups {
}
}
addBubble(bubble: HTMLDivElement, message: MyMessage, reverse: boolean) {
addBubble(bubble: HTMLElement, message: MyMessage, reverse: boolean) {
//return;
const timestamp = message.date;
@ -41,7 +42,7 @@ export default class BubbleGroups {
}
// try to find added
//this.removeBubble(message.mid);
this.removeBubble(bubble);
const insertObject = {bubble, mid, timestamp};
if(this.bubbles.length) {
@ -105,28 +106,24 @@ export default class BubbleGroups {
//console.log('[BUBBLE]: addBubble', bubble, message.mid, fromId, reverse, group);
if(mid > 0) {
let insertIndex = 0;
for(; insertIndex < this.bubbles.length; ++insertIndex) {
if(this.bubbles[insertIndex].mid < mid) {
break;
}
const bubbleGroup = {timestamp, fromId, mid: message.mid, group};
let insertIndex = 0;
for(; insertIndex < this.bubbles.length; ++insertIndex) {
if(this.bubbles[insertIndex].mid < mid) {
break;
}
this.bubbles.splice(insertIndex, 0, {timestamp, fromId, mid: message.mid, group});
} else {
this.bubbles.unshift({timestamp, fromId, mid: message.mid, group});
}
//this.bubbles.push({timestamp, fromId, mid: message.mid, group});
this.bubbles.splice(insertIndex, 0, {timestamp, fromId, mid: message.mid, group});
this.updateGroup(group);
this.detailsMap.set(bubble, bubbleGroup);
}
setClipIfNeeded(bubble: HTMLDivElement, remove = false) {
/* setClipIfNeeded(bubble: HTMLDivElement, remove = false) {
//console.log('setClipIfNeeded', bubble, remove);
const className = bubble.className;
if(className.includes('is-message-empty')/* && !className.includes('is-reply') */
&& (className.includes('photo') || className.includes('video'))) {
if(className.includes('is-message-empty') && (className.includes('photo') || className.includes('video'))) {
let container = bubble.querySelector('.bubble__media-container') as SVGSVGElement;
//console.log('setClipIfNeeded', bubble, remove, container);
if(!container) return;
@ -171,7 +168,7 @@ export default class BubbleGroups {
});
} catch(err) {}
}
}
} */
updateGroup(group: Group) {
/* if(this.updateRAFs.has(group)) {
@ -192,25 +189,25 @@ export default class BubbleGroups {
if(group.length === 1) {
first.classList.add('is-group-first', 'is-group-last');
this.setClipIfNeeded(first);
//this.setClipIfNeeded(first);
return;
} else {
first.classList.remove('is-group-last');
first.classList.add('is-group-first');
this.setClipIfNeeded(first, true);
//this.setClipIfNeeded(first, true);
}
const length = group.length - 1;
for(let i = 1; i < length; ++i) {
const bubble = group[i].bubble;
bubble.classList.remove('is-group-last', 'is-group-first');
this.setClipIfNeeded(bubble, true);
//this.setClipIfNeeded(bubble, true);
}
const last = group[group.length - 1].bubble;
last.classList.remove('is-group-first');
last.classList.add('is-group-last');
this.setClipIfNeeded(last);
//this.setClipIfNeeded(last);
//}));
}
@ -224,6 +221,7 @@ export default class BubbleGroups {
cleanup() {
this.bubbles = [];
this.groups = [];
this.detailsMap.clear();
/* for(let value of this.updateRAFs.values()) {
window.cancelAnimationFrame(value);
}

View File

@ -152,12 +152,9 @@ export default class ChatBubbles {
const message = this.chat.getMessage(mid);
//bubble.remove();
this.bubbleGroups.removeBubble(bubble, message.mid);
this.setBubblePosition(bubble, message, false);
//this.log('history_update', this.bubbles[mid], mid, message);
this.bubbleGroups.addBubble(bubble, message, false);
if(this.scrollingToNewBubble) {
this.scrollToNewLastBubble();
}
@ -268,8 +265,6 @@ export default class ChatBubbles {
//});
bubble.dataset.mid = '' + mid;
this.bubbleGroups.removeBubble(bubble, tempId);
}
if(this.unreadOut.has(tempId)) {
@ -423,7 +418,7 @@ export default class ChatBubbles {
const msgIdsByPeer = e;
if(!(this.peerId in msgIdsByPeer)) return;
const msgIds = msgIdsByPeer[this.peerId];
const msgIds = (msgIdsByPeer[this.peerId] as number[]).slice().sort((a, b) => b - a);
this.renderNewMessagesByIds(msgIds);
});
@ -1147,7 +1142,7 @@ export default class ChatBubbles {
this.firstUnreadBubble = null;
}
this.bubbleGroups.removeBubble(bubble, mid);
this.bubbleGroups.removeBubble(bubble);
if(this.unreadedObserver) {
this.unreadedObserver.unobserve(bubble);
}
@ -1691,6 +1686,8 @@ export default class ChatBubbles {
dateMessage.container.append(bubble);
}
}
this.bubbleGroups.addBubble(bubble, message, reverse);
}
// * will change .cleaned in cleanup() and new instance will be created
@ -1779,16 +1776,8 @@ export default class ChatBubbles {
bubble.remove(); // * for positionElementByIndex
} */
if(!sameMid || updatePosition) {
this.bubbleGroups.removeBubble(bubble, originalMid);
}
if(!sameMid) {
delete this.bubbles[originalMid];
if(!updatePosition) {
this.bubbleGroups.addBubble(bubble, message, reverse);
}
}
//bubble.innerHTML = '';
@ -1815,9 +1804,7 @@ export default class ChatBubbles {
if(updatePosition) {
this.renderMessagesQueue(message, bubble, reverse, loadPromises);
if(!message.pFlags.is_single) { // * Ignore 'Discussion started'
this.bubbleGroups.addBubble(bubble, message, reverse);
} else {
if(message.pFlags.is_single) { // * Ignore 'Discussion started'
bubble.classList.add('is-group-last');
}
}
@ -2452,13 +2439,7 @@ export default class ChatBubbles {
bubble.classList.add(isOut ? 'is-out' : 'is-in');
if(updatePosition) {
if(!isThreadStarter) {
this.bubbleGroups.addBubble(bubble, message, reverse);
}
this.renderMessagesQueue(message, bubble, reverse, loadPromises);
} else {
this.bubbleGroups.updateGroupByMessageId(message.mid);
}
if(withReplies) {

View File

@ -215,7 +215,7 @@ export default class ChatContextMenu {
icon: 'link',
text: 'Copy Link',
onClick: this.onCopyLinkClick,
verify: () => this.appPeersManager.isChannel(this.peerId)
verify: () => this.appPeersManager.isChannel(this.peerId) && !this.message.pFlags.is_outgoing
}, {
icon: 'pin',
text: 'Pin',

View File

@ -3313,7 +3313,7 @@ export class AppMessagesManager {
} as any,
id: this.generateMessageId(message.id, true),
date: message.date,
from_id: message.from_id,
from_id: {_: 'peerUser', user_id: 0}/* message.from_id */,
peer_id: message.peer_id,
action: {
_: 'messageActionCustomAction',

View File

@ -156,7 +156,7 @@ export class AppStateManager extends EventListenerBase<{
if(state) {
if(state.version !== STATE_VERSION) {
state = copy(STATE_INIT);
} else if((state.stateCreatedTime + REFRESH_EVERY) < time/* && false */) {
} else if((state.stateCreatedTime + REFRESH_EVERY) < time/* || true *//* && false */) {
if(DEBUG) {
this.log('will refresh state', state.stateCreatedTime, time);
}

View File

@ -118,7 +118,9 @@ export class AppUsersManager {
this.pushContact(userId);
});
this.contactsFillPromise = Promise.resolve(this.contactsList);
if(this.contactsList.size) {
this.contactsFillPromise = Promise.resolve(this.contactsList);
}
}
});
}
@ -605,7 +607,7 @@ export class AppUsersManager {
return apiManager.invokeApi('contacts.getTopPeers', {
correspondents: true,
offset: 0,
limit: 30,
limit: 15,
hash: 0,
}).then((result) => {
let peerIds: number[] = [];

View File

@ -16,7 +16,7 @@ import { addPadding, bytesFromBigInt } from '../mtproto/bin_utils';
import { bytesToWordss, bytesFromWordss, bytesToHex, bytesFromHex } from '../../helpers/bytes';
import { nextRandomInt } from '../../helpers/random';
export function longToBytes(sLong: string) {
export function longToBytes(sLong: string): Array<number> {
/* let perf = performance.now();
for(let i = 0; i < 1000000; ++i) {
bytesFromWords({words: longToInts(sLong), sigBytes: 8}).reverse();

View File

@ -157,12 +157,12 @@ export class ApiManager {
// mtpGetNetworker
public getNetworker(dcId: number, options: InvokeApiOptions = {}): Promise<MTPNetworker> {
const connectionType: ConnectionType = options.fileDownload ? 'download' : (options.fileUpload ? 'upload' : 'client');
//const connectionType: ConnectionType = 'client';
//const connectionType: ConnectionType = options.fileDownload ? 'download' : (options.fileUpload ? 'upload' : 'client');
const connectionType: ConnectionType = 'client';
/// #if MTPROTO_HTTP_UPLOAD
// @ts-ignore
const transportType: TransportType = connectionType === 'upload' && isSafari ? 'https' : 'websocket';
const transportType: TransportType = connectionType === 'upload' && isSafari && false ? 'https' : 'websocket';
//const transportType: TransportType = connectionType !== 'client' ? 'https' : 'websocket';
/// #else
// @ts-ignore
@ -183,7 +183,7 @@ export class ApiManager {
}
const networkers = cache[dcId];
if(networkers.length >= /* 1 */(connectionType === 'client' || transportType === 'https' ? 1 : (connectionType === 'download' ? 3 : 3))) {
if(networkers.length >= /* 1 */(connectionType === 'client' || transportType === 'https' ? 1 : (connectionType === 'download' ? 3 : 1))) {
let i = networkers.length - 1, found = false;
for(; i >= 0; --i) {
if(networkers[i].isOnline) {

View File

@ -34,3 +34,33 @@ export const Database = {
export const DEBUG = process.env.NODE_ENV !== 'production' || Modes.debug;
export const MOUNT_CLASS_TO: any = DEBUG ? (typeof(window) !== 'undefined' ? window : self) : null;
export const superDebug = (object: any, key: string) => {
var d = object[key];
var beforeStr = '', afterStr = '';
for(var r of d) {
beforeStr += r.before.hex + '\n';
afterStr += r.after.hex + '\n';
}
beforeStr = beforeStr.trim();
afterStr = afterStr.trim();
//var beforeStr = d.map(r => r.before.hex).join('\n');
//var afterStr = d.map(r => r.after.hex).join('\n');
var dada = (name: string, str: string) => {
var a = document.createElement('a');
a.target = '_blank';
a.download = name + '.txt';
a.href = URL.createObjectURL(new Blob([str], {
type: 'text/plain'
}));
document.body.append(a);
a.click();
};
dada(key + '_' + 'before', beforeStr);
dada(key + '_' + 'after', afterStr);
}
MOUNT_CLASS_TO && (MOUNT_CLASS_TO.superDebug = superDebug);

View File

@ -124,7 +124,7 @@ export class ApiManagerProxy extends CryptoWorkerMethods {
//const worker = window;
worker.addEventListener('message', (e) => {
if(!this.worker) {
this.worker = worker;
this.worker = worker as any;
this.log('set webWorker');
if(USE_WORKER_AS_WORKER) {

View File

@ -100,8 +100,9 @@ export default class MTPNetworker {
/// #endif
private seqNo: number = 0;
private prevSessionId: Array<number> = [];
private sessionId: Array<number> = [];
private prevSessionId: Uint8Array;
private sessionId: Uint8Array;
private serverSalt: Uint8Array;
private lastResendReq: {
req_msg_id: string,
@ -119,9 +120,12 @@ export default class MTPNetworker {
private tt = 0;
//public onConnectionStatusChange: (online: boolean) => void;
private debugRequests: Array<{before: Uint8Array, after: Uint8Array}> = [];
constructor(public dcId: number, private authKey: number[], private authKeyId: Uint8Array,
private serverSalt: number[], private transport: MTTransport, options: InvokeApiOptions = {}) {
serverSalt: number[], private transport: MTTransport, options: InvokeApiOptions = {}) {
this.authKeyUint8 = convertToUint8Array(this.authKey);
this.serverSalt = convertToUint8Array(serverSalt);
this.isFileUpload = !!options.fileUpload;
this.isFileDownload = !!options.fileDownload;
@ -179,7 +183,7 @@ export default class MTPNetworker {
public updateSession() {
this.seqNo = 0;
this.prevSessionId = this.sessionId;
this.sessionId = [...new Uint8Array(8).randomize()];
this.sessionId = new Uint8Array(8).randomize();
}
public updateSentMessage(sentMessageId: string) {
@ -231,7 +235,7 @@ export default class MTPNetworker {
const message = {
msg_id: messageId,
seq_no: seqNo,
body: serializer.getBytes()
body: serializer.getBytes(true)
};
if(Modes.debug) {
@ -250,7 +254,7 @@ export default class MTPNetworker {
const message = {
msg_id: messageId,
seq_no: seqNo,
body: serializer.getBytes()
body: serializer.getBytes(true)
};
if(Modes.debug) {
@ -335,7 +339,7 @@ export default class MTPNetworker {
}
private sendPingDelayDisconnect = () => {
if(this.pingPromise) return;
if(this.pingPromise || true) return;
if(!this.isOnline) {
if((this.transport as Socket).connected) {
@ -667,28 +671,28 @@ export default class MTPNetworker {
this.scheduleRequest(delay);
}
// * correct, fully checked
public async getMsgKey(dataWithPadding: ArrayBuffer, isOut: boolean) {
const authKey = this.authKeyUint8;
const x = isOut ? 0 : 8
const msgKeyLargePlain = bufferConcat(authKey.subarray(88 + x, 88 + x + 32), dataWithPadding);
const msgKeyLargePlain = bufferConcat(this.authKeyUint8.subarray(88 + x, 88 + x + 32), dataWithPadding);
const msgKeyLarge = await CryptoWorker.sha256Hash(msgKeyLargePlain);
const msgKey = new Uint8Array(msgKeyLarge).subarray(8, 24);
return msgKey;
};
// * correct, fully checked
public getAesKeyIv(msgKey: Uint8Array | number[], isOut: boolean): Promise<[Uint8Array, Uint8Array]> {
const authKey = this.authKeyUint8;
const x = isOut ? 0 : 8;
const sha2aText = new Uint8Array(52);
const sha2bText = new Uint8Array(52);
const promises: Array<Promise<number[]>> = [];
sha2aText.set(msgKey, 0);
sha2aText.set(authKey.subarray(x, x + 36), 16);
sha2aText.set(this.authKeyUint8.subarray(x, x + 36), 16);
promises.push(CryptoWorker.sha256Hash(sha2aText));
sha2bText.set(authKey.subarray(40 + x, 40 + x + 36), 0);
sha2bText.set(this.authKeyUint8.subarray(40 + x, 40 + x + 36), 0);
sha2bText.set(msgKey, 36);
promises.push(CryptoWorker.sha256Hash(sha2bText));
@ -900,7 +904,7 @@ export default class MTPNetworker {
return {
bytes: encryptedBytes,
msgKey: msgKey
msgKey
};
}
@ -934,6 +938,14 @@ export default class MTPNetworker {
data.storeInt(message.body.length, 'message_data_length');
data.storeRawBytes(message.body, 'message_data');
const des = new TLDeserialization(data.getBuffer().slice(16));
const desSalt = des.fetchLong();
const desSessionId = des.fetchLong();
if(!this.isOnline) {
this.log.error('trying to send message when offline', message, new Uint8Array(des.buffer), desSalt, desSessionId);
}
/* const messageDataLength = message.body.length;
let canBeLength = 0; // bytes
canBeLength += 8;
@ -950,13 +962,15 @@ export default class MTPNetworker {
} */
const paddingLength = (16 - (data.offset % 16)) + 16 * (1 + nextRandomInt(5));
const padding = [...new Uint8Array(paddingLength).randomize()];
const padding = (message as any).padding || new Uint8Array(paddingLength).randomize().fill(0);
/* const padding = [167, 148, 207, 226, 86, 192, 193, 57, 124, 153, 174, 145, 159, 1, 5, 70, 127, 157,
51, 241, 46, 85, 141, 212, 139, 234, 213, 164, 197, 116, 245, 70, 184, 40, 40, 201, 233, 211, 150,
94, 57, 84, 1, 135, 108, 253, 34, 139, 222, 208, 71, 214, 90, 67, 36, 28, 167, 148, 207, 226, 86, 192, 193, 57, 124, 153, 174, 145, 159, 1, 5, 70, 127, 157,
51, 241, 46, 85, 141, 212, 139, 234, 213, 164, 197, 116, 245, 70, 184, 40, 40, 201, 233, 211, 150,
94, 57, 84, 1, 135, 108, 253, 34, 139, 222, 208, 71, 214, 90, 67, 36, 28].slice(0, paddingLength); */
(message as any).padding = padding;
const dataWithPadding = bufferConcat(dataBuffer, padding);
// this.log('Adding padding', dataBuffer, padding, dataWithPadding)
// this.log('auth_key_id', bytesToHex(self.authKeyID))
@ -969,6 +983,7 @@ export default class MTPNetworker {
this.log('Send encrypted: body length:', (message.body as ArrayBuffer).byteLength, paddingLength, dataWithPadding);
} */
// * full next block is correct
return this.getEncryptedMessage(dataWithPadding).then((encryptedResult) => {
/* if(DEBUG) {
this.log('Got encrypted out message', encryptedResult);
@ -983,10 +998,12 @@ export default class MTPNetworker {
const requestData = request.getBytes(true);
//if(message.fileUpload) {
if(this.isFileNetworker) {
//this.log('Send encrypted: requestData length:', requestData.length, requestData.length % 16, paddingLength % 16, paddingLength, data.offset, encryptedResult.msgKey.length % 16, encryptedResult.bytes.length % 16);
//this.log('Send encrypted: messageId:', message.msg_id, requestData.length);
//}
//this.log('Send encrypted:', message, new Uint8Array(bufferConcat(des.buffer, padding)), requestData, this.serverSalt.hex, this.sessionId.hex/* new Uint8Array(des.buffer) */);
this.debugRequests.push({before: new Uint8Array(bufferConcat(des.buffer, padding)), after: requestData});
}
return requestData;
});
@ -1137,7 +1154,7 @@ export default class MTPNetworker {
return {
response,
messageId,
sessionId: sessionId,
sessionId,
seqNo
};
});
@ -1151,7 +1168,7 @@ export default class MTPNetworker {
['dc' + this.dcId + '_server_salt']: bytesToHex(serverSalt)
});
this.serverSalt = serverSalt;
this.serverSalt = new Uint8Array(serverSalt);
}
// ! таймаут очень сильно тормозит скорость работы сокета (даже нулевой)

View File

@ -1,5 +1,6 @@
import { bytesToHex } from '../../helpers/bytes';
import { bigint, bigStringInt, isObject } from './bin_utils';
import { MOUNT_CLASS_TO } from './mtproto_config';
/// #if MTPROTO_WORKER
// @ts-ignore
import { gzipUncompress } from '../crypto/crypto_utils';
@ -789,5 +790,6 @@ class TLDeserialization {
}
}
MOUNT_CLASS_TO && (MOUNT_CLASS_TO.TLDeserialization = TLDeserialization);
MOUNT_CLASS_TO && (MOUNT_CLASS_TO.TLSerialization = TLSerialization);
export { TLDeserialization, TLSerialization };

View File

@ -4,13 +4,15 @@ export class IntermediatePacketCodec implements Codec {
public tag = 0xee;
public obfuscateTag = new Uint8Array([this.tag, this.tag, this.tag, this.tag]);
//private lol = 0;
public encodePacket(data: Uint8Array) {
if((data.length % 4) !== 0) {
console.error('Encode error!', data.length, data);
}
const len = data.length;
const header = new Uint8Array(new Uint32Array([len]).buffer);
const header = new Uint8Array(new Int32Array([/* ++this.lol >= 25 ? 0x80000001 : */len]).buffer);
//console.log('got nobody cause im braindead', header, len, /* data, */data.buffer.byteLength === data.length);
return header.concat(data);

View File

@ -95,8 +95,8 @@ export default class Obfuscation {
/* this.enc = new aesjs.ModeOfOperation.ctr(encKey, new aesjs.Counter(encIv as any));
this.dec = new aesjs.ModeOfOperation.ctr(decKey, new aesjs.Counter(decIv as any)); */
/* console.log('encKey', encKey, encIv);
console.log('decKey', decKey, decIv); */
console.log('encKey', encKey.hex, encIv.hex);
console.log('decKey', decKey.hex, decIv.hex);
this.encNew = new CTR(encKey, encIv);
this.decNew = new CTR(decKey, decIv);

View File

@ -34,6 +34,8 @@ export default class Socket extends MTTransport {
//private lol: Uint8Array[] = [];
//private dd: () => void;
//private debugPayloads: MTPNetworker['debugRequests'] = [];
constructor(dcId: number, url: string, logSuffix: string, public retryTimeout: number) {
super(dcId, url);
@ -220,6 +222,8 @@ export default class Socket extends MTTransport {
const enc = this.obfuscation.encode(toEncode);
//this.log('send after obf:', enc.hex);
//this.debugPayloads.push({before: body.slice(), after: enc});
this.debug && this.log.debug('-> body length to send:', enc.length, this.ws.bufferedAmount);
/* if(this.ws.bufferedAmount) {
this.log.error('bufferedAmount:', this.ws.bufferedAmount);

92
t.py
View File

@ -1,8 +1,16 @@
from telethon import TelegramClient, events, sync
from telethon.network import ConnectionTcpAbridged
from telethon.network.mtprotostate import MTProtoState
from telethon.crypto.authkey import AuthKey
from telethon.network.connection.tcpobfuscated import ConnectionTcpObfuscated
from telethon.network.connection.tcpintermediate import IntermediatePacketCodec
import sys
print(sys.path)
import struct
import logging
logging.basicConfig(level=logging.DEBUG)
# import sys
# print(sys.path)
# These example values won't work. You must get your own api_id and
# api_hash from https://my.telegram.org, under API Development.
@ -11,4 +19,82 @@ api_hash = '452b0359b988148995f22ff0f4229750'
client = TelegramClient('session_name', api_id, api_hash, connection=ConnectionTcpAbridged)
#client.start()
client.connect()
#client.connect()
authKey = AuthKey(bytes.fromhex("2c45ed62da34d41bc82b6ef660760d3b01cb80028b0c363633f1c40ec80d5f4ad9d4ddee646736b6a9465fd7258b8936dedde6a86c69b4a03277ed6c543ccfd31fa8fa9f9449d686b4822b54542c91255231ecfa6aed4a9896cdcc3d5491e7f1b7529cbff75597f262813ced7eb3b5ac1e7e6794ca7aa00c5ea4d1d714e7c276fb63e5f8a1742cf4707eace8a6ec2ed4dcceceabdc7fc92bd099c811931e0da337f1b0271d8ba33e7a4a22fa1e4a913d8220edb0dd3a810ed408f03fe5d7f2d5dd43db4272d314b20a22376c03b0a05423600f4d012e7196ce9b3c45b28a75dbfa053222f5c86d1c6a706a2fd68ca270fd9a6449df1fa15a21837d34f4cf4e95"))
state = MTProtoState(authKey, loggers=client._log)
state.salt = struct.unpack('q', bytes.fromhex("d1b37eebb4997ca9"))[0]
state.id = struct.unpack('q', bytes.fromhex("502368ee46d39313"))[0]
#print(state.id)
""" with open('file_part.txt', 'r') as file:
data = file.read()
messageData = bytes.fromhex(data)
#messageData = bytes.fromhex("146f8265b5342560290000000c000000ec77be7aaa40f90300000000")
#print(data)
encrypted = state.encrypt_message_data(messageData)
file.close()
with open('file_part_encrypted.txt', 'w') as file:
file.write(encrypted.hex())
file.close()
#print(encrypted) """
ConnectionTcpObfuscated.packet_codec = IntermediatePacketCodec
connection = ConnectionTcpObfuscated("", "", 2, loggers=client._log)
obfuscatedIo = connection.obfuscated_io(connection)
#print(encrypted)
#exit()
#encoded = connection.packet_codec.encode_packet(connection.packet_codec, bytes.fromhex("deadbeef"))
#print(encoded)
#exit()
name = "debugPayloads"
#name = "debugRequests"
with open(name + '_before.txt', 'r') as fileBefore:
data = fileBefore.read()
lines = data.splitlines()
with open(name + '_after.txt', 'r') as fileAfter:
data = fileAfter.read()
linesAfter = data.splitlines()
length = len(lines)
for i in range(length):
print("processing line %i, of %i", i, length)
lineBefore = lines[i]
lineAfter = linesAfter[i]
messageData = bytes.fromhex(lineBefore)
messageData = connection.packet_codec.encode_packet(connection.packet_codec, messageData)
encrypted = obfuscatedIo._encrypt.encrypt(messageData)
#encrypted = state.encrypt_message_data(messageData) # need to comment padding inside
#print(encrypted.hex())
#print(len(encrypted.hex()))
#print(len(lineAfter))
#print(encrypted.hex() == lineAfter)
#break
if encrypted.hex() != lineAfter:
print("lol")
with open('difference_before.txt', 'w') as file:
file.write(lineAfter)
file.close()
with open('difference_after.txt', 'w') as file:
file.write(encrypted.hex())
file.close()
#print(encrypted.hex())
#print(lineAfter)
exit()

83
t2.py Normal file
View File

@ -0,0 +1,83 @@
from telethon import TelegramClient, events, sync
from telethon.network import ConnectionTcpAbridged
from telethon.network.mtprotostate import MTProtoState
from telethon.crypto.authkey import AuthKey
import struct
import logging
logging.basicConfig(level=logging.DEBUG)
# import sys
# print(sys.path)
# These example values won't work. You must get your own api_id and
# api_hash from https://my.telegram.org, under API Development.
api_id = 1025907
api_hash = '452b0359b988148995f22ff0f4229750'
client = TelegramClient('session_name', api_id, api_hash, connection=ConnectionTcpAbridged)
#client.start()
#client.connect()
authKey = AuthKey(bytes.fromhex("2c45ed62da34d41bc82b6ef660760d3b01cb80028b0c363633f1c40ec80d5f4ad9d4ddee646736b6a9465fd7258b8936dedde6a86c69b4a03277ed6c543ccfd31fa8fa9f9449d686b4822b54542c91255231ecfa6aed4a9896cdcc3d5491e7f1b7529cbff75597f262813ced7eb3b5ac1e7e6794ca7aa00c5ea4d1d714e7c276fb63e5f8a1742cf4707eace8a6ec2ed4dcceceabdc7fc92bd099c811931e0da337f1b0271d8ba33e7a4a22fa1e4a913d8220edb0dd3a810ed408f03fe5d7f2d5dd43db4272d314b20a22376c03b0a05423600f4d012e7196ce9b3c45b28a75dbfa053222f5c86d1c6a706a2fd68ca270fd9a6449df1fa15a21837d34f4cf4e95"))
state = MTProtoState(authKey, loggers=client._log)
state.salt = struct.unpack('q', bytes.fromhex("d1b37eebb4997ca9"))[0]
state.id = struct.unpack('q', bytes.fromhex("502368ee46d39313"))[0]
print(state.id)
""" with open('file_part.txt', 'r') as file:
data = file.read()
messageData = bytes.fromhex(data)
#messageData = bytes.fromhex("146f8265b5342560290000000c000000ec77be7aaa40f90300000000")
#print(data)
encrypted = state.encrypt_message_data(messageData)
file.close()
with open('file_part_encrypted.txt', 'w') as file:
file.write(encrypted.hex())
file.close()
#print(encrypted) """
with open('debugRequests_before.txt', 'r') as fileBefore:
data = fileBefore.read()
lines = data.splitlines()
with open('debugRequests_after.txt', 'r') as fileAfter:
data = fileAfter.read()
linesAfter = data.splitlines()
length = len(lines)
for i in range(length):
print("processing line %i, of %i", i, length)
lineBefore = lines[i]
messageData = bytes.fromhex(lineBefore)
encrypted = state.encrypt_message_data(messageData) # need to comment padding inside
lineAfter = linesAfter[i]
#print(len(encrypted.hex()))
#print(len(lineAfter))
#print(encrypted.hex() == lineAfter)
#break
if encrypted.hex() != lineAfter:
print("lol")
with open('difference_before.txt', 'w') as file:
file.write(lineAfter)
file.close()
with open('difference_after.txt', 'w') as file:
file.write(encrypted.hex())
file.close()
#print(encrypted.hex())
#print(lineAfter)
exit()

View File

@ -8,8 +8,8 @@ const fs = require('fs');
const allowedIPs = ['194.58.97.147', '195.66.140.39', '127.0.0.1', '176.100.8.254'];
const devMode = process.env.NODE_ENV !== 'production';
const useLocal = false;
const useLocalNotLocal = false;
const useLocal = true;
const useLocalNotLocal = true;
if(devMode) {
console.log('DEVMODE IS ON!');
@ -103,7 +103,7 @@ module.exports = {
allowedHosts: useLocal ? undefined : [
'tweb.enko.club'
],
host: useLocalNotLocal ? '192.168.93.183' : (useLocal ? undefined : '0.0.0.0'),
host: useLocalNotLocal ? '192.168.93.209' : (useLocal ? undefined : '0.0.0.0'),
public: useLocal ? undefined : 'tweb.enko.club',
//host: '192.168.0.105', // '0.0.0.0'
//host: 'tweb.enko.club', // '0.0.0.0'