Add "Last Platform" and "Last Watched" to user data tables

This commit is contained in:
JonnyWong16 2015-08-19 21:08:17 -07:00 committed by Jonathan Wong
parent 968d213b97
commit 5168d76e86
5 changed files with 226 additions and 43 deletions

View file

@ -24,13 +24,11 @@ user_ip_table_options = {
}, },
"searchable": false, "searchable": false,
"width": "15%", "width": "15%",
"className": "no-wrap" "className": "no-wrap hidden-xs"
}, },
{ {
"targets": [1], "targets": [1],
"data":"ip_address", "data": "ip_address",
"width": "15%",
"className": "modal-control no-wrap",
"createdCell": function (td, cellData, rowData, row, col) { "createdCell": function (td, cellData, rowData, row, col) {
if (cellData) { if (cellData) {
if (isPrivateIP(cellData)) { if (isPrivateIP(cellData)) {
@ -46,31 +44,59 @@ user_ip_table_options = {
$(td).html('n/a'); $(td).html('n/a');
} }
}, },
"width": "15%" "width": "15%",
"className": "no-wrap modal-control-ip"
}, },
{ {
"targets": [2], "targets": [2],
"data":"play_count", "data":"platform",
"width": "10%", "createdCell": function (td, cellData, rowData, row, col) {
"className": "hidden-xs" if (cellData) {
$(td).html('<a href="#" data-target="#info-modal" data-toggle="modal"><i class="fa fa-lg fa-info-circle"></i>&nbsp' + cellData + '</a>');
} else {
$(td).html('n/a');
}
},
"width": "15%",
"className": "no-wrap hidden-md hidden-sm hidden-xs modal-control"
}, },
{ {
"targets": [3], "targets": [3],
"data":"platform", "data":"last_watched",
"width": "15%", "createdCell": function (td, cellData, rowData, row, col) {
"className": "hidden-xs" if (cellData !== '') {
if (rowData['media_type'] === 'movie' || rowData['media_type'] === 'episode') {
var transcode_dec = '';
if (rowData['video_decision'] === 'transcode') {
transcode_dec = '<i class="fa fa-server"></i>&nbsp';
}
$(td).html('<div><div style="float: left;"><a href="info?source=history&item_id=' + rowData['id'] + '">' + cellData + '</a></div><div style="float: right; text-align: right; padding-right: 5px;">' + transcode_dec + '<i class="fa fa-video-camera"></i></div></div>');
} else if (rowData['media_type'] === 'track') {
$(td).html('<div><div style="float: left;">' + cellData + '</div><div style="float: right; text-align: right; padding-right: 5px;"><i class="fa fa-music"></i></div></div>');
} else if (rowData['media_type']) {
$(td).html('<a href="info?item_id=' + rowData['id'] + '">' + cellData + '</a>');
} else {
$(td).html('n/a');
}
}
},
"className": "hidden-sm hidden-xs"
}, },
{ {
"targets": [4], "targets": [4],
"data":"last_watched", "data":"play_count",
"width": "30%", "searchable": false,
"className": "hidden-sm hidden-xs" "width": "10%"
} }
], ],
"drawCallback": function (settings) { "drawCallback": function (settings) {
// Jump to top of page // Jump to top of page
// $('html,body').scrollTop(0); // $('html,body').scrollTop(0);
$('#ajaxMsg').fadeOut(); $('#ajaxMsg').fadeOut();
// Create the tooltips.
$('.info-modal').each(function () {
$(this).tooltip();
});
}, },
"preDrawCallback": function(settings) { "preDrawCallback": function(settings) {
var msg = "<div class='msg'><i class='fa fa-refresh fa-spin'></i>&nbspFetching rows...</div>"; var msg = "<div class='msg'><i class='fa fa-refresh fa-spin'></i>&nbspFetching rows...</div>";
@ -83,6 +109,25 @@ $('#user_ip_table').on('mouseenter', 'td.modal-control span', function () {
}); });
$('#user_ip_table').on('click', 'td.modal-control', function () { $('#user_ip_table').on('click', 'td.modal-control', function () {
var tr = $(this).parents('tr');
var row = user_ip_table.row(tr);
var rowData = row.data();
function showStreamDetails() {
$.ajax({
url: 'get_stream_data',
data: { row_id: rowData['id'], user: rowData['friendly_name'] },
cache: false,
async: true,
complete: function (xhr, status) {
$("#info-modal").html(xhr.responseText);
}
});
}
showStreamDetails();
});
$('#user_ip_table').on('click', 'td.modal-control-ip', function () {
var tr = $(this).parents('tr'); var tr = $(this).parents('tr');
var row = user_ip_table.row( tr ); var row = user_ip_table.row( tr );
var rowData = row.data(); var rowData = row.data();

View file

@ -45,6 +45,7 @@ users_list_table_options = {
$(td).html(cellData); $(td).html(cellData);
} }
}, },
"width": "15%"
}, },
{ {
"targets": [2], "targets": [2],
@ -57,24 +58,70 @@ users_list_table_options = {
} }
}, },
"searchable": false, "searchable": false,
"className": "hidden-xs", "width": "15%",
"className": "no-wrap hidden-xs"
}, },
{ {
"targets": [3], "targets": [3],
"data": "ip_address", "data": "ip_address",
"render": function ( data, type, full ) { "createdCell": function (td, cellData, rowData, row, col) {
if (data) { if (cellData) {
return data; if (isPrivateIP(cellData)) {
if (cellData != '') {
$(td).html(cellData);
} else { } else {
return "n/a"; $(td).html('n/a');
}
} else {
$(td).html('<a href="javascript:void(0)" data-toggle="modal" data-target="#ip-info-modal"><span data-toggle="ip-tooltip" data-placement="left" title="IP Address Info" id="ip-info"><i class="fa fa-map-marker"></i></span>&nbsp' + cellData + '</a>');
}
} else {
$(td).html('n/a');
} }
}, },
"className": "hidden-xs", "width": "15%",
"className": "no-wrap hidden-md hidden-sm hidden-xs modal-control-ip"
}, },
{ {
"targets": [4], "targets": [4],
"data":"platform",
"createdCell": function (td, cellData, rowData, row, col) {
if (cellData) {
$(td).html('<a href="#" data-target="#info-modal" data-toggle="modal"><i class="fa fa-lg fa-info-circle"></i>&nbsp' + cellData + '</a>');
} else {
$(td).html('n/a');
}
},
"width": "15%",
"className": "no-wrap hidden-md hidden-sm hidden-xs modal-control"
},
{
"targets": [5],
"data":"last_watched",
"createdCell": function (td, cellData, rowData, row, col) {
if (cellData !== '') {
if (rowData['media_type'] === 'movie' || rowData['media_type'] === 'episode') {
var transcode_dec = '';
if (rowData['video_decision'] === 'transcode') {
transcode_dec = '<i class="fa fa-server"></i>&nbsp';
}
$(td).html('<div><div style="float: left;"><a href="info?source=history&item_id=' + rowData['id'] + '">' + cellData + '</a></div><div style="float: right; text-align: right; padding-right: 5px;">' + transcode_dec + '<i class="fa fa-video-camera"></i></div></div>');
} else if (rowData['media_type'] === 'track') {
$(td).html('<div><div style="float: left;">' + cellData + '</div><div style="float: right; text-align: right; padding-right: 5px;"><i class="fa fa-music"></i></div></div>');
} else if (rowData['media_type']) {
$(td).html('<a href="info?item_id=' + rowData['id'] + '">' + cellData + '</a>');
} else {
$(td).html('n/a');
}
}
},
"className": "hidden-sm hidden-xs"
},
{
"targets": [6],
"data": "plays", "data": "plays",
"searchable": false "searchable": false,
"width": "10%"
} }
], ],
@ -82,9 +129,55 @@ users_list_table_options = {
// Jump to top of page // Jump to top of page
//$('html,body').scrollTop(0); //$('html,body').scrollTop(0);
$('#ajaxMsg').fadeOut(); $('#ajaxMsg').fadeOut();
// Create the tooltips.
$('.info-modal').each(function () {
$(this).tooltip();
});
}, },
"preDrawCallback": function(settings) { "preDrawCallback": function(settings) {
var msg = "<div class='msg'><i class='fa fa-refresh fa-spin'></i>&nbspFetching rows...</div>"; var msg = "<div class='msg'><i class='fa fa-refresh fa-spin'></i>&nbspFetching rows...</div>";
showMsg(msg, false, false, 0) showMsg(msg, false, false, 0)
} }
} }
$('#users_list_table').on('click', 'td.modal-control', function () {
var tr = $(this).parents('tr');
var row = users_list_table.row(tr);
var rowData = row.data();
function showStreamDetails() {
$.ajax({
url: 'get_stream_data',
data: { row_id: rowData['id'], user: rowData['friendly_name'] },
cache: false,
async: true,
complete: function (xhr, status) {
$("#info-modal").html(xhr.responseText);
}
});
}
showStreamDetails();
});
$('#users_list_table').on('click', 'td.modal-control-ip', function () {
var tr = $(this).parents('tr');
var row = users_list_table.row(tr);
var rowData = row.data();
function getUserLocation(ip_address) {
if (isPrivateIP(ip_address)) {
return "n/a"
} else {
$.ajax({
url: 'get_ip_address_details',
data: { ip_address: ip_address },
async: true,
complete: function (xhr, status) {
$("#ip-info-modal").html(xhr.responseText);
}
});
}
}
getUserLocation(rowData['ip_address']);
});

View file

@ -131,11 +131,11 @@ from plexpy import helpers
<table id="user_ip_table" class="display" width="100%"> <table id="user_ip_table" class="display" width="100%">
<thead> <thead>
<tr> <tr>
<th align="left">Last seen</th> <th align="left">Last Seen</th>
<th align="left">IP Address</th> <th align="left">IP Address</th>
<th align="left">Play Count</th> <th align="left">Last Platform</th>
<th align="left">Platform (Last Seen)</th>
<th align="left">Last Watched</th> <th align="left">Last Watched</th>
<th align="left">Play Count</th>
</tr> </tr>
</thead> </thead>
</table> </table>
@ -181,7 +181,6 @@ from plexpy import helpers
</tbody> </tbody>
</table> </table>
</div> </div>
<div id="info-modal" class="modal fade" tabindex="-1" role="dialog" aria-labelledby="info-modal"></div>
</div> </div>
</div> </div>
</div> </div>
@ -224,7 +223,9 @@ from plexpy import helpers
</div> </div>
</div> </div>
</div> </div>
<div id="ip-info-modal" class="modal fade" tabindex="-1" role="dialog" aria-labelledby="ip-info-modal"> <div class="modal fade" id="info-modal" tabindex="-1" role="dialog" aria-labelledby="info-modal">
</div>
<div class="modal fade" id="ip-info-modal" tabindex="-1" role="dialog" aria-labelledby="ip-info-modal">
</div> </div>
</div> </div>
<footer></footer> <footer></footer>

View file

@ -23,12 +23,18 @@
<th align="left" id="friendly_name">User</th> <th align="left" id="friendly_name">User</th>
<th align="left" id="last_seen">Last Seen</th> <th align="left" id="last_seen">Last Seen</th>
<th align="left" id="last_known_ip">Last Known IP</th> <th align="left" id="last_known_ip">Last Known IP</th>
<th align="left" id="last_platform">Last Platform</th>
<th align="left" id="last_watched">Last Watched</th>
<th align="left" id="total_plays">Total Plays</th> <th align="left" id="total_plays">Total Plays</th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
</tbody> </tbody>
</table> </table>
<div class="modal fade" id="info-modal" tabindex="-1" role="dialog" aria-labelledby="info-modal">
</div>
<div class="modal fade" id="ip-info-modal" tabindex="-1" role="dialog" aria-labelledby="ip-info-modal">
</div>
</div> </div>
</div> </div>

View file

@ -24,13 +24,19 @@ class Users(object):
def get_user_list(self, kwargs=None): def get_user_list(self, kwargs=None):
data_tables = datatables.DataTables() data_tables = datatables.DataTables()
columns = ['users.user_id as user_id', columns = ['session_history.id',
'users.user_id as user_id',
'users.custom_avatar_url as thumb', 'users.custom_avatar_url as thumb',
'(case when users.friendly_name is null then users.username else \ '(case when users.friendly_name is null then users.username else \
users.friendly_name end) as friendly_name', users.friendly_name end) as friendly_name',
'MAX(session_history.started) as last_seen', 'MAX(session_history.started) as last_seen',
'session_history.ip_address as ip_address', 'session_history.ip_address as ip_address',
'COUNT(session_history.id) as plays', 'COUNT(session_history.id) as plays',
'session_history.player as platform',
'session_history_metadata.full_title as last_watched',
'session_history_metadata.media_type',
'session_history.rating_key as rating_key',
'session_history_media_info.video_decision',
'users.username as user' 'users.username as user'
] ]
try: try:
@ -38,9 +44,15 @@ class Users(object):
columns=columns, columns=columns,
custom_where=[], custom_where=[],
group_by=['users.user_id'], group_by=['users.user_id'],
join_types=['LEFT OUTER JOIN'], join_types=['LEFT OUTER JOIN',
join_tables=['session_history'], 'LEFT OUTER JOIN',
join_evals=[['session_history.user_id', 'users.user_id']], 'LEFT OUTER JOIN'],
join_tables=['session_history',
'session_history_metadata',
'session_history_media_info'],
join_evals=[['session_history.user_id', 'users.user_id'],
['session_history.id', 'session_history_metadata.id'],
['session_history.id', 'session_history_media_info.id']],
kwargs=kwargs) kwargs=kwargs)
except: except:
logger.warn("Unable to execute database query.") logger.warn("Unable to execute database query.")
@ -59,10 +71,16 @@ class Users(object):
else: else:
user_thumb = item['thumb'] user_thumb = item['thumb']
row = {"plays": item['plays'], row = {"id": item['id'],
"plays": item['plays'],
"last_seen": item['last_seen'], "last_seen": item['last_seen'],
"friendly_name": item["friendly_name"], "friendly_name": item['friendly_name'],
"ip_address": item["ip_address"], "ip_address": item['ip_address'],
"platform": item['platform'],
"last_watched": item['last_watched'],
"media_type": item['media_type'],
"rating_key": item['rating_key'],
"video_decision": item['video_decision'],
"thumb": user_thumb, "thumb": user_thumb,
"user": item["user"], "user": item["user"],
"user_id": item['user_id'] "user_id": item['user_id']
@ -81,13 +99,22 @@ class Users(object):
def get_user_unique_ips(self, kwargs=None, custom_where=None): def get_user_unique_ips(self, kwargs=None, custom_where=None):
data_tables = datatables.DataTables() data_tables = datatables.DataTables()
columns = ['session_history.started as last_seen', # Change custom_where column name due to ambiguous column name after JOIN
custom_where[0][0] = 'custom_user_id' if custom_where[0][0] == 'user_id' else custom_where[0][0]
columns = ['session_history.id',
'session_history.started as last_seen',
'session_history.ip_address as ip_address', 'session_history.ip_address as ip_address',
'COUNT(session_history.id) as play_count', 'COUNT(session_history.id) as play_count',
'session_history.player as platform', 'session_history.player as platform',
'session_history_metadata.full_title as last_watched', 'session_history_metadata.full_title as last_watched',
'session_history_metadata.media_type',
'session_history.rating_key as rating_key',
'session_history_media_info.video_decision',
'session_history.user as user', 'session_history.user as user',
'session_history.user_id as user_id' 'session_history.user_id as custom_user_id',
'(case when users.friendly_name is null then users.username else \
users.friendly_name end) as friendly_name'
] ]
try: try:
@ -95,9 +122,15 @@ class Users(object):
columns=columns, columns=columns,
custom_where=custom_where, custom_where=custom_where,
group_by=['ip_address'], group_by=['ip_address'],
join_types=['JOIN'], join_types=['JOIN',
join_tables=['session_history_metadata'], 'JOIN',
join_evals=[['session_history.id', 'session_history_metadata.id']], 'JOIN'],
join_tables=['users',
'session_history_metadata',
'session_history_media_info'],
join_evals=[['session_history.user_id', 'users.user_id'],
['session_history.id', 'session_history_metadata.id'],
['session_history.id', 'session_history_media_info.id']],
kwargs=kwargs) kwargs=kwargs)
except: except:
logger.warn("Unable to execute database query.") logger.warn("Unable to execute database query.")
@ -111,11 +144,16 @@ class Users(object):
rows = [] rows = []
for item in results: for item in results:
row = {"last_seen": item['last_seen'], row = {"id": item['id'],
"last_seen": item['last_seen'],
"ip_address": item['ip_address'], "ip_address": item['ip_address'],
"play_count": item['play_count'], "play_count": item['play_count'],
"platform": item['platform'], "platform": item['platform'],
"last_watched": item['last_watched'] "last_watched": item['last_watched'],
"media_type": item['media_type'],
"rating_key": item['rating_key'],
"video_decision": item['video_decision'],
"friendly_name": item['friendly_name']
} }
rows.append(row) rows.append(row)