Отслеживать информацию о посетителях по каждому запросу

Используя веб-формы Asp.net, я хочу отслеживать информацию о посетителях так же, как это делает Google Analytics. Конечно, я могу использовать Google Analytic для этой цели, но я хочу знать, как я могу добиться того же с Asp.net 3.5 и SQL Server 2008.

Я хочу сохранить IP, страну, URL Referrer посетителя, разрешение на каждом запросе страницы, кроме обратной передачи. Я ожидаю 50 000+ посещений каждый день.,

Главное, что я хочу сделать так, чтобы он не блокировал текущий запрос.

т.е. в общем случае это происходит, когда мы сохраняем данные в дБ, текущий запрос останавливается на определенном состоянии вызова SP и продвигается, когда он завершает выполнение оператора SP или tsql. Я хочу следовать подходу "вставь и забудь". Он должен вставляться в фоновом режиме, когда я передаю параметр определенному событию или функции.

Я нашел ниже альтернативы для этого:
1. PageAsynchTask
2. BeginExecuteNonQuery
3. Метод Jquery Post и Webservice (но я не уверен в этом, и мне интересно, как мне это сделать)

Я надеюсь, что упомянул мою проблему правильно.

Кто-нибудь может сказать мне, какой из них лучше подходит? Также дайте мне знать, если у вас есть другие идеи или лучший подход, чем перечисленные. Ваша помощь будет по достоинству оценена.

4 ответа

Решение

Проблемы с любым фоновым потоком на стороне сервера - каждый запрос будет занимать два потока. Один для обслуживания запроса ASP.NET и один для регистрации материала, который вы хотите зарегистрировать. Таким образом, у вас возникают проблемы с масштабируемостью из-за исчерпания потоков ASP.NET. И регистрация каждого запроса в базе данных является большой нет, нет.

Лучше всего просто записывать файлы журналов, используя высокопроизводительную библиотеку журналов. Библиотеки журналов высоко оптимизированы для многопоточных журналов. Они не производят вызовы ввода / вывода при каждом вызове. Журналы хранятся в буфере памяти и периодически очищаются. Вы должны использовать EntLib или Log4net для регистрации.

Вы можете использовать HttpModule, который перехватывает каждый GET, POST, а затем внутри HttpModule вы можете проверить, является ли Request.Url aspx или нет. Затем вы можете прочитать Request.Headers["__ASYNCPOST"] и посмотреть, является ли оно "true", что означает, что это асинхронное обновление UpdatePanel. Если все эти условия выполняются, вы просто регистрируете запрос в файле журнала, в котором хранятся

Вы можете получить IP клиента от:

HttpContext.Current.Request.UserHostAddress; 
or 
HttpContext.Current.Request.ServerVariables["REMOTE_ADDR"];

Чтобы получить IP-адрес компьютера, а не прокси, используйте следующий код

HttpContext.Current.Request.ServerVariables["HTTP_X_FORWARDED_FOR"];

Однако вы не можете получить страну. Вам нужно будет зарегистрировать IP-адрес в своих файлах журналов, а затем обработать файлы журналов, используя какое-либо консольное приложение или задание, которое определит страну с IP-адресом. Вам нужно получить некоторую базу данных IP->Country, чтобы сделать работу. Я использовал http://www.maxmind.com/app/geoip_country раньше.

Для размера экрана вам придется положиться на некоторый JavaScript. Используйте javascript на каждой странице, который определяет размер экрана на стороне клиента и сохраняет его в файле cookie.

var screenW = 640, screenH = 480;
if (parseInt(navigator.appVersion)>3) {
 screenW = screen.width;
 screenH = screen.height;
}
else if (navigator.appName == "Netscape" 
    && parseInt(navigator.appVersion)==3
    && navigator.javaEnabled()
   ) 
{
 var jToolkit = java.awt.Toolkit.getDefaultToolkit();
 var jScreenSize = jToolkit.getScreenSize();
 screenW = jScreenSize.width;
 screenH = jScreenSize.height;
}

Как только вы сохраните его в файле cookie (я не показал этот код), вы можете прочитать размеры экрана из модуля HttpModule, используя Request.Cookies, а затем зарегистрировать его в файле журнала.

Таким образом, это дает вам решение для регистрации IP, размера экрана, поиска страны по IP и фильтрации асинхронной обратной передачи UpdatePanel из регистрации.

Это дает вам полное решение проблемы?

Если говорить о стороне сервера, если вы работаете на IIS и вам не нужна абсолютная информация в реальном времени, я рекомендую вам использовать журналы IIS.

Нет ничего быстрее, чем это, так как он был оптимизирован для производительности, начиная с IIS 1.0

Вы можете добавить свою собственную информацию в эти журналы ( HttpRequest.AppendToLog), они имеют стандартный формат, есть API, если вы хотите делать с ним пользовательские вещи (но вы все равно можете использовать анализатор текста, если хотите), и есть множество бесплатных инструментов, например, Microsoft Log Parser, который может передавать данные в базу данных SQL (среди прочих).

Первый подход выглядит хорошо. (И я рекомендую это.) Но у этого есть 2 недостатка:

  1. Запрос будет блокироваться до тех пор, пока задача не будет завершена (или не прервана по тайм-ауту)
  2. Вы должны будете зарегистрировать свою задачу на каждой странице.

Второй подход выглядит неудобно и может привести к ошибкам. (Вы должны следить за ситуацией, когда ваша страница отображается быстрее, чем обрабатывается ваш запрос. Я не уверен, что произойдет, если ваш запрос не будет завершен, когда ваш объект страницы будет уничтожен и GC продолжит выполнение finalize()... но ничего хорошего, я полагаю. Вы можете избежать этого, ожидая IAsyncResult.IsCompleted после рендера, но это неудобно.)

Третий метод совершенно неправильный. Вы должны начать свою регистрацию на стороне сервера при обработке запроса, который вы собираетесь регистрировать. Но вы все равно можете вызвать веб-сервис со стороны сервера. (Или выигрышный сервис).

Лично я хотел бы реализовать ведение журнала в BeginRequest, чтобы избежать дублирования кода, но вам нужен IsPostback... Тем не менее, может быть обходной путь.

Hii, вы можете запустить асинхронный запрос и не ждать ответа. здесь у меня есть некоторый реализованный код..

для этого вам нужно создать веб-сервис для выполнения операций с базой данных или использовать его для всей обработки событий.

со стороны сервера вы должны вызывать веб-сервис асинхронно, как это

Объявить частного делегата

private delegate void ReEntryDelegate(long CaseID, string MessageText);

Теперь метод будет содержать вызов веб-службы, подобный этому

WebServiceTest.Notification service = new WebServiceTest.Notification();
IAsyncResult handle;
ReEntryDelegate objAscReEntry = new ReEntryDelegate(service.ReEntryNotifications);
handle = objAscReEntry.BeginInvoke(CaseID, MessageText, null, null);
break;

И значения переменных здесь будут переданы методом (CaseID,MessageText)

Надеюсь, это ясно для вас

Всего наилучшего

Другие вопросы по тегам