const axios = require('axios')
const qs = require('querystring')
const setCookie = require('set-cookie-parser')
const fs = require('fs')
const FormData = require('form-data')
const EksiGuest = require('./EksiGuest')
const { URLS } = require('./constants')
const { TITLE_TYPES } = require('./enums')
const { tags, trashEntries, debeEntries, createEntry } = require('./lib')
const {
EntryForMember,
UserForMember,
TitleCollection,
EntryCollection,
SearchResults,
TrashEntry,
TagForMember,
DraftEntry
} = require('./models')
/**
* Eksi Sozluk member class.
*
* @augments EksiGuest
*/
class EksiMember extends EksiGuest {
/**
* Username.
*
* @type {string}
*/
username
/**
* Is new message available?
*
* @type {boolean}
*/
isNewMessageAvailable
/**
* Is new event available?
*
* @type {boolean}
*/
isNewEventAvailable
/**
* Create an Eksi Sozluk member session.
*
* @param {object} httpClient Axios HTTP client.
* @param {string} cookies Cookies in string.
*/
constructor (httpClient, cookies) {
super(httpClient)
this.cookies = cookies
}
/**
* Retrieve the user.
*
* @returns {Promise} Promise.
*/
retrieve () {
return new Promise((resolve, reject) => {
axios({
url: URLS.BASE,
method: 'GET',
headers: {
cookie: this.cookies
}
}).then(res => {
const newMessageRegex = /href="\/mesaj"\n*\s*class="new-update"/g
const isNewMessageAvailable = newMessageRegex.test(res.data)
this.isNewMessageAvailable = isNewMessageAvailable
const newEventRegex = /title="olaylar olaylar"\n*\s*class="new-update"/g
const isNewEventAvailable = newEventRegex.test(res.data)
this.isNewEventAvailable = isNewEventAvailable
const usernameRegex = new RegExp(
'(?<=a href="/biri/)(.*)(?=" title=")',
'u'
)
const username = usernameRegex.exec(res.data)[0]
this.username = username
resolve()
})
})
}
/**
* Search things.
*
* @param {string} text Search text.
* @returns {Promise.<SearchResults>} A promise for the search results.
*/
async search (text) {
const results = new SearchResults(this._request, text, this.cookies)
await results.retrieve()
return results
}
/**
* Check if unread message exist.
*
* @returns {Promise.<boolean>} New message available or not.
*/
isNewMessageExist () {
return new Promise((resolve, reject) => {
axios({
url: URLS.BASE,
method: 'GET',
headers: {
cookie: this.cookies
}
}).then(res => {
const regex = /href="\/mesaj"\n*\s*class="new-update"/g
const isNewMessageAvailable = regex.test(res.data)
this.isNewMessageAvailable = isNewMessageAvailable
resolve(isNewMessageAvailable)
})
})
}
/**
* Check if unread event exist.
*
* @returns {Promise.<boolean>} New event available or not.
*/
isNewEventExist () {
return new Promise((resolve, reject) => {
axios({
url: URLS.BASE,
method: 'GET',
headers: {
cookie: this.cookies
}
}).then(res => {
const regex = /title="olaylar olaylar"\n*\s*class="new-update"/g
const isNewEventAvailable = regex.test(res.data)
this.isNewEventAvailable = isNewEventAvailable
resolve(isNewEventAvailable)
})
})
}
/**
* Checking if your email address is in change status.
*
* @returns {Promise.<boolean>} Email address waiting for changing or not.
*/
isEmailAddressInChangeStatus () {
return new Promise((resolve, reject) => {
axios({
url: URLS.SETTINGS_EMAIL,
method: 'GET',
headers: {
cookie: this.cookies
}
}).then(res => {
const isEmailAddressInChangingStatus = res.data.includes(
'değişikliği iptal et'
)
resolve(isEmailAddressInChangingStatus)
})
})
}
/**
* Cancel the email address change.
*
* @returns {Promise.<boolean>} A promise for cancel the email address change.
*/
cancelEmailAddressChange () {
return new Promise((resolve, reject) => {
axios({
url: URLS.SETTINGS_EMAIL,
method: 'GET',
headers: {
cookie: this.cookies
}
})
.then(res => {
// validate page is in email change status
const isWaitingForCancelEmail = res.data.includes(
'değişikliği iptal et'
)
if (!isWaitingForCancelEmail) {
return reject(new Error('Not waiting for change email.'))
}
return res
})
.then(res => {
// parse csrf token
const csrfRegex = new RegExp(
'(?<=input name="__RequestVerificationToken" type="hidden" value=")(.*)(?=" />)',
'u'
)
const csrfToken = csrfRegex.exec(res.data)[0]
const cookies = setCookie.parse(res.headers['set-cookie'], {
map: true
})
const csrfTokenInCookies = cookies.__RequestVerificationToken.value
return { csrfToken, csrfTokenInCookies }
})
.then(async ({ csrfToken, csrfTokenInCookies }) => {
// cancel email address change
const _res = await axios({
url: URLS.SETTINGS_CANCEL_UPDATE_EMAIL,
method: 'POST',
headers: {
Cookie: `__RequestVerificationToken=${csrfTokenInCookies}; ${this.cookies}`
},
data: qs.stringify({
__RequestVerificationToken: csrfToken
})
})
return _res
})
.then(res => {
const isSucc = res.status === 200
if (!isSucc) {
return reject(new Error('An unknown error occurred.'))
}
resolve()
})
})
}
/**
* Change your password.
*
* @param {string} currPassword Your current password.
* @param {string} newPassword A new password.
* @returns {Promise.<boolean>} A promise for change password.
*/
changePassword (currPassword, newPassword) {
return new Promise((resolve, reject) => {
axios({
url: URLS.SETTINGS_PASSWORD,
method: 'GET',
headers: {
cookie: this.cookies
}
})
.then(res => {
// parse csrf token
const csrfRegex = new RegExp(
'(?<=input name="__RequestVerificationToken" type="hidden" value=")(.*)(?=" />)',
'u'
)
const csrfToken = csrfRegex.exec(res.data)[0]
const cookies = setCookie.parse(res.headers['set-cookie'], {
map: true
})
const csrfTokenInCookies = cookies.__RequestVerificationToken.value
return { csrfToken, csrfTokenInCookies }
})
.then(async ({ csrfToken, csrfTokenInCookies }) => {
// change password
const _res = await axios({
url: URLS.SETTINGS_PASSWORD,
method: 'POST',
headers: {
Cookie: `__RequestVerificationToken=${csrfTokenInCookies}; ${this.cookies}`
},
data: qs.stringify({
__RequestVerificationToken: csrfToken,
OldPassword: currPassword,
Password: newPassword,
PasswordConfirm: newPassword
}),
validateStatus: status => {
// successful response returns 404 status so accept 4xx responses
return status >= 200 && status < 500
}
})
return _res
})
.then(res => {
const isSucc =
res.data.includes('şifreniz güncellendi') && res.status === 404
const isCurrPasswordWrong = res.data.includes(
'şu anki şifrenizi yanlış girdiniz'
)
const isUnknownErr = res.data.includes(
'<title>büyük başarısızlıklar sözkonusu - ekşi sözlük</title>'
)
const isTooManyRequest = res.status === 429
if (isTooManyRequest) {
return reject(new Error('Too many request for changing password.'))
}
if (isCurrPasswordWrong) {
return reject(new Error('Current password is wrong.'))
}
if (!isSucc || isUnknownErr) {
return reject(new Error('An unknown error occurred.'))
}
// update cookies with the new token
const cookies = setCookie.parse(res.headers['set-cookie'], {
map: true
})
const newToken = cookies.a.value
this.cookies = `a=${newToken}`
resolve()
})
})
}
/**
* Delete your account.
*
* @param {string} password Your current password.
* @param {boolean} [hideEntries=false] Hide your entries.
* @returns {Promise} Promise.
*/
deleteAccount (password, hideEntries = false) {
return new Promise((resolve, reject) => {
axios({
url: URLS.SETTINGS_DELETE_ACCOUNT,
method: 'POST',
headers: {
Cookie: this.cookies
},
data: qs.stringify({
Password: password,
HideEntries: hideEntries
})
}).then(res => {
resolve()
})
})
}
/**
* Change login username.
*
* @param {string} newUsername Your new login username.
* @param {string} password Your current password.
* @returns {Promise} Promise.
*/
changeLoginUsername (newUsername, password) {
return new Promise((resolve, reject) => {
axios({
url: URLS.SETTINGS_CHANGE_USERNAME,
method: 'GET',
headers: {
cookie: this.cookies
}
})
.then(res => {
// parse csrf token
const csrfRegex = new RegExp(
'(?<=input name="__RequestVerificationToken" type="hidden" value=")(.*)(?=" />)',
'u'
)
const csrfToken = csrfRegex.exec(res.data)[0]
const cookies = setCookie.parse(res.headers['set-cookie'], {
map: true
})
const csrfTokenInCookies = cookies.__RequestVerificationToken.value
return { csrfToken, csrfTokenInCookies }
})
.then(async ({ csrfToken, csrfTokenInCookies }) => {
// change password
const _res = await axios({
url: URLS.SETTINGS_CHANGE_USERNAME,
method: 'POST',
headers: {
Cookie: `__RequestVerificationToken=${csrfTokenInCookies}; ${this.cookies}`
},
data: qs.stringify({
__RequestVerificationToken: csrfToken,
Password: password,
NewLoginName: newUsername
}),
validateStatus: status => {
// successful response returns 404 status so accept 4xx responses
return status >= 200 && status < 500
}
})
return _res
})
.then(res => {
const isSucc =
res.data.includes(
'giriş için kullandığınız kullanıcı adını güncelledik'
) && res.status === 404
if (!isSucc) {
return reject(new Error('An unknown error occurred.'))
}
resolve()
})
})
}
/**
* Change your email address.
*
* @param {string} currEmailAddress Your current email address.
* @param {string} newEmailAddress A new email address.
* @param {string} password Your current password.
* @returns {Promise} A promise for change email address.
*/
changeEmailAddress (currEmailAddress, newEmailAddress, password) {
return new Promise((resolve, reject) => {
axios({
url: URLS.SETTINGS_EMAIL,
method: 'GET',
headers: {
cookie: this.cookies
}
})
.then(res => {
// validate page is in email change status
const isWaitingForCancelEmail = res.data.includes(
'değişikliği iptal et'
)
if (isWaitingForCancelEmail) {
return reject(new Error('Email address already changed.'))
}
return res
})
.then(res => {
// parse csrf token
const csrfRegex = new RegExp(
'(?<=input name="__RequestVerificationToken" type="hidden" value=")(.*)(?=" />)',
'u'
)
const csrfToken = csrfRegex.exec(res.data)[0]
const cookies = setCookie.parse(res.headers['set-cookie'], {
map: true
})
const csrfTokenInCookies = cookies.__RequestVerificationToken.value
return { csrfToken, csrfTokenInCookies }
})
.then(async ({ csrfToken, csrfTokenInCookies }) => {
// change password
const _res = await axios({
url: URLS.SETTINGS_EMAIL,
method: 'POST',
headers: {
Cookie: `__RequestVerificationToken=${csrfTokenInCookies}; ${this.cookies}`
},
data: qs.stringify({
__RequestVerificationToken: csrfToken,
CurrentEmail: currEmailAddress,
Password: password,
NewEmail: newEmailAddress,
ConfirmNewEmail: newEmailAddress
})
})
return _res
})
.then(res => {
const isSucc =
res.data.includes('değişikliği iptal et') && res.status === 200
if (!isSucc) {
return reject(new Error('An unknown error occurred.'))
}
resolve()
})
})
}
/**
* Create backup.
*
* @returns {Promise} A promise for create backup.
*/
createBackup () {
return new Promise((resolve, reject) => {
axios({
url: URLS.SETTINGS_CREATE_BACKUP,
method: 'POST',
responseType: 'arraybuffer',
headers: {
Cookie: this.cookies
},
validateStatus: status => {
// for catch five minute error
return status >= 200 && status < 500
}
}).then(res => {
const isSucc = res.status === 200
const isFiveMinuteError =
res.data.includes('yedekleri 5 dakika ara ile verebiliyoruz.') &&
res.status === 404
if (isFiveMinuteError) {
return reject(new Error('You can create backup every 5 minutes.'))
}
if (!isSucc) {
return reject(new Error('An unknown error occurred.'))
}
resolve(res.data)
})
})
}
/**
* Pin an entry to the profile.
*
* @param {number} entryId Entry ID which user owns.
* @returns {Promise} Promise.
*/
pinEntry (entryId) {
return new Promise((resolve, reject) => {
axios
.post(URLS.PIN, qs.stringify({ entryId }), {
headers: {
cookie: this.cookies,
'x-requested-with': 'XMLHttpRequest'
}
})
.then(res => {
if (res.data.Success) {
resolve()
} else {
reject(new Error('It is not your entry or entry is not yours.'))
}
})
})
}
/**
* Remove pin from profile.
*
* @returns {Promise} Promise.
*/
removePin () {
return new Promise((resolve, reject) => {
axios
.post(URLS.PIN_REMOVE, null, {
headers: {
cookie: this.cookies,
'x-requested-with': 'XMLHttpRequest'
}
})
.then(res => {
if (res.data !== true) {
return reject(new Error('No pinned entry found.'))
}
resolve()
})
})
}
/**
* Create entry.
*
* @param {string} title Title.
* @param {string} content Entry content.
* @param {object} options Parameters that user can specify.
* @param {boolean} [options.saveAsDraft=false] Save as draft.
* @returns {Promise.<(EntryForMember|DraftEntry)>} Created entry.
*/
async createEntry (title, content, options = {}) {
return await createEntry(
this._request,
title,
content,
options,
this.cookies
)
}
/**
* Fetch entry by id.
*
* @param {number} entryId Entry Id.
* @returns {Promise.<EntryForMember>} A promise for the entry.
*/
async entryById (entryId) {
const entry = new EntryForMember(this._request, entryId, this.cookies)
await entry.retrieve()
return entry
}
/**
* Fetch entries.
*
* @param {string} title Title itself.
* @param {object} options Parameters that user can specify.
* @param {number} [options.page=1] Page number.
* @returns {Promise.<EntryCollection>} A promise for the entries.
*/
async entries (title, options = {}) {
const _options = {
...options,
cookies: this.cookies
}
const collection = new EntryCollection(this._request, title, _options)
await collection.retrieve()
return collection
}
/**
* Fetch user.
*
* @param {string} username Entry Id.
* @returns {Promise.<UserForMember>} A promise for the entry.
*/
async user (username) {
const user = new UserForMember(this._request, username, this.cookies)
await user.retrieve()
return user
}
/**
* Fetch the user profile.
*
* @returns {Promise.<UserForMember>} A promise for the entry.
*/
async me () {
const user = new UserForMember(this._request, this.username, this.cookies)
await user.retrieve()
return user
}
/**
* Fetch today entries.
*
* @param {object} options Parameters that user can specify.
* @param {number} [options.page=1] Page number.
* @returns {Promise.<TitleCollection>} A promise for the titles of today.
*/
async today (options = {}) {
const target = '/basliklar/bugun'
const _options = {
...options,
cookies: this.cookies
}
const collection = new TitleCollection(this._request, target, _options)
await collection.retrieve()
return collection
}
/**
* Fetch rookie entries.
*
* @param {object} options Parameters that user can specify.
* @param {number} [options.page=1] Page number.
* @returns {Promise.<TitleCollection>} A promise for the rookie titles.
*/
async rookieTitles (options = {}) {
const target = '/basliklar/caylaklar'
const _options = { ...options, cookies: this.cookies }
const collection = new TitleCollection(this._request, target, _options)
await collection.retrieve()
return collection
}
/**
* Fetch events.
*
* @returns {Promise.<TitleCollection>} A promise for the titles of events.
*/
async events () {
const target = '/basliklar/olay'
const _options = { defaultEntryCount: 0, cookies: this.cookies }
const collection = new TitleCollection(this._request, target, _options)
await collection.retrieve()
return collection
}
/**
* Fetch draft entries.
*
* @param {object} options Parameters that user can specify.
* @param {number} [options.page=1] Page number.
* @returns {Promise.<TitleCollection>} A promise for the titles of drafts.
*/
async drafts (options = {}) {
const target = '/basliklar/kenar'
const _options = {
...options,
type: TITLE_TYPES.DRAFT,
cookies: this.cookies
}
const collection = new TitleCollection(this._request, target, _options)
await collection.retrieve()
return collection
}
/**
* Fetch followed user titles.
*
* @param {object} options Parameters that user can specify.
* @param {number} [options.page=1] Page number.
* @returns {Promise.<TitleCollection>} A promise for the followed user titles.
*/
async followedUserTitles (options = {}) {
const target = '/basliklar/takipentry'
const _options = {
...options,
type: TITLE_TYPES.FOLLOWED_USER,
cookies: this.cookies
}
const collection = new TitleCollection(this._request, target, _options)
await collection.retrieve()
return collection
}
/**
* Fetch followed user favorite entries.
*
* @param {object} options Parameters that user can specify.
* @param {number} [options.page=1] Page number.
* @returns {Promise.<TitleCollection>} A promise for the followed user titles.
*/
async followedUserFavoriteEntries (options = {}) {
const target = '/basliklar/takipfav'
const _options = {
...options,
type: TITLE_TYPES.FOLLOWED_USER_FAVORITE_ENTRY,
cookies: this.cookies
}
const collection = new TitleCollection(this._request, target, _options)
await collection.retrieve()
return collection
}
/**
* Fetch tags.
*
* @returns {Promise.Array<TagForMember>} A promise for the tags.
*/
async tags () {
return await tags(this._request, this.cookies)
}
/**
* Fetch yesterday's top entries.
*
* @returns {Promise.Array<EntryForMember>} A promise for the yesterday's top entries.
*/
async debeEntries () {
return await debeEntries(this._request, this.cookies)
}
/**
* Fetch trash entries.
*
* @param {object} options Parameters that user can specify.
* @param {number} [options.page=1] Page number.
* @returns {Promise.Array<TrashEntry>} A promise for the trash entries.
*/
async trashEntries (options = {}) {
return await trashEntries(this._request, this.cookies, options)
}
/**
* Empty trash.
*
* @returns {Promise} Promise.
*/
emptyTrash () {
return new Promise((resolve, reject) => {
axios({
url: URLS.TRASH,
method: 'GET',
headers: {
cookie: this.cookies
}
})
.then(res => {
// parse csrf token
const csrfRegex = new RegExp(
'(?<=input name="__RequestVerificationToken" type="hidden" value=")(.*)(?=" />)',
'u'
)
const csrfToken = csrfRegex.exec(res.data)[0]
return csrfToken
})
.then(async csrfToken => {
// empty trash
const _res = await axios({
url: URLS.TRASH_EMPTY,
method: 'POST',
headers: {
cookie: this.cookies
},
data: qs.stringify({
__RequestVerificationToken: csrfToken
})
})
return _res
})
.then(res => {
if (
res.data.includes(
'<title>büyük başarısızlıklar sözkonusu - ekşi sözlük</title>'
)
) {
reject(new Error('Unknown Error'))
} else {
resolve()
}
})
})
}
/**
* @typedef UploadedImage
* @property {string} url Image url.
* @property {string} key Image key.
*/
/**
* Upload image.
*
* @param {string} imagePath Image file path.
* @returns {Promise.<UploadedImage>} Promise.
*/
uploadImage (imagePath) {
return new Promise((resolve, reject) => {
const data = new FormData()
data.append('file', fs.createReadStream(imagePath))
axios
.post(`${URLS.BASE}/img/func/${this.username}`, data, {
headers: {
'x-requested-with': 'XMLHttpRequest',
Cookie: this.cookies,
...data.getHeaders()
}
})
.then(response => {
// handle errors
if (!response.data.Success) {
return reject(new Error(response.data.Result))
}
resolve({
url: response.data.Result,
key: response.data.ImageKey
})
})
})
}
/**
* Remove image with given image key.
*
* @param {string} imageKey Image key.
* @returns {Promise} Promise.
*/
deleteImage (imageKey) {
const data = qs.stringify({
imageKey: imageKey,
reasonCode: 'osel' // don't know why.
})
return new Promise((resolve, reject) => {
axios
.post(`${URLS.BASE}/img/func/sil`, data, {
headers: {
'x-requested-with': 'XMLHttpRequest',
'Content-Type': 'application/x-www-form-urlencoded',
Cookie: this.cookies
}
})
.then(response => {
// handle errors
if (!response.data.Success) {
return reject(new Error('An unknown error occurred.'))
}
resolve()
})
})
}
}
module.exports = EksiMember