mirror of
https://github.com/Tautulli/Tautulli.git
synced 2025-07-07 05:31:15 -07:00
Add "Last Platform" and "Last Watched" to user data tables
This commit is contained in:
parent
968d213b97
commit
5168d76e86
5 changed files with 226 additions and 43 deletions
|
@ -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> ' + 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> ';
|
||||||
|
}
|
||||||
|
$(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> Fetching rows...</div>";
|
var msg = "<div class='msg'><i class='fa fa-refresh fa-spin'></i> Fetching 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();
|
||||||
|
|
|
@ -34,7 +34,7 @@ users_list_table_options = {
|
||||||
{
|
{
|
||||||
"targets": [1],
|
"targets": [1],
|
||||||
"data": "friendly_name",
|
"data": "friendly_name",
|
||||||
"createdCell": function (td, cellData, rowData, row, col) {
|
"createdCell": function (td, cellData, rowData, row, col) {
|
||||||
if (cellData !== '') {
|
if (cellData !== '') {
|
||||||
if (rowData['user_id'] > 0) {
|
if (rowData['user_id'] > 0) {
|
||||||
$(td).html('<a href="user?user_id=' + rowData['user_id'] + '">' + cellData + '</a>');
|
$(td).html('<a href="user?user_id=' + rowData['user_id'] + '">' + cellData + '</a>');
|
||||||
|
@ -45,6 +45,7 @@ users_list_table_options = {
|
||||||
$(td).html(cellData);
|
$(td).html(cellData);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"width": "15%"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"targets": [2],
|
"targets": [2],
|
||||||
|
@ -57,34 +58,126 @@ 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 {
|
||||||
|
$(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> ' + cellData + '</a>');
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
return "n/a";
|
$(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> ' + 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> ';
|
||||||
|
}
|
||||||
|
$(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%"
|
||||||
|
}
|
||||||
|
|
||||||
],
|
],
|
||||||
"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> Fetching rows...</div>";
|
var msg = "<div class='msg'><i class='fa fa-refresh fa-spin'></i> Fetching 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']);
|
||||||
|
});
|
|
@ -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>
|
||||||
|
|
|
@ -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>
|
||||||
|
|
||||||
|
|
|
@ -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)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue