Last active 1531452378

MiniDLNA Python REST API server and associated Tkinter GUI for reading from it https://github.com/blha303/PythonMiniDLNARestAPI

Revision 01494d3633fcf002336604ea6d394f37b0cb9b24

README.md Raw

I made this because I really wanted to be able to search my media server with Python. I need to write a longer README here.

gui.py Raw
1# Run this on a client machine (not the media server)
2# This is my first GUI!
3
4# Copyright 2013 Steven Smith (blha303). All Rights Reserved.
5# New BSD license
6# http://www.opensource.org/licenses/BSD-3-Clause
7
8from Tkinter import *
9import tkMessageBox
10import requests
11import json
12
13hostname = "home.blha303.com.au"
14url = "http://%s:%s/api/DETAILS" % (hostname, port if port else "5000")
15fields = ['ARTIST', 'ID', 'PATH', 'SIZE', 'TIMESTAMP', 'TITLE', 'DURATION', 'BITRATE',
16 'SAMPLERATE', 'CREATOR', 'ALBUM', 'GENRE', 'COMMENT', 'CHANNELS', 'DISC',
17 'TRACK', 'DATE', 'RESOLUTION', 'THUMBNAIL', 'ALBUM_ART', 'ROTATION',
18 'DLNA_PN', 'MIME']
19
20def search(term, name='artist'):
21 url = "http://home.blha303.com.au:5000/api/DETAILS"
22 headers = {'Content-Type': 'application/json'}
23 filters = [dict(name=name.upper(), op='like', val=term)]
24 params = dict(q=json.dumps(dict(filters=filters)))
25 response = requests.get(url, params=params, headers=headers)
26 assert response.status_code == 200
27 return response.json()
28
29master = Tk()
30windows = PanedWindow()
31windows.pack(fill=BOTH, expand=1)
32textbox = Text(windows)
33windows.add(textbox)
34rightside = PanedWindow(windows, orient=VERTICAL)
35windows.add(rightside)
36top = Entry(rightside)
37rightside.add(top)
38middle = Listbox(rightside)
39x = 1
40for i in fields:
41 middle.insert(x, i)
42 x += 1
43
44rightside.add(middle)
45
46def buttonCallback():
47 term = top.get()
48 name = middle.get(middle.curselection() if middle.curselection() else 0)
49 data = search(term, name=name)
50 textbox.delete(1.0, END)
51 if data['num_results'] == 0:
52 textbox.insert(INSERT, str(data))
53 return
54 else:
55 output = []
56 for i in data['objects']:
57 if str(i['PATH']) != "None":
58 output.append("PATH: " + str(i['PATH']))
59 output.append(" ID: " + str(i['ID']))
60 output.append(" SIZE: " + str(i['SIZE']))
61 output.append(" TIMESTAMP: " + str(i['TIMESTAMP']))
62 output.append(" TITLE: " + str(i['TITLE']))
63 output.append(" DURATION: " + str(i['DURATION']))
64 output.append(" BITRATE: " + str(i['BITRATE']))
65 output.append(" SAMPLERATE: " + str(i['SAMPLERATE']))
66 output.append(" CREATOR: " + str(i['CREATOR']))
67 output.append(" ARTIST: " + str(i['ARTIST']))
68 output.append(" ALBUM: " + str(i['ALBUM']))
69 output.append(" GENRE: " + str(i['GENRE']))
70 output.append(" COMMENT: " + str(i['COMMENT']))
71 output.append(" CHANNELS: " + str(i['CHANNELS']))
72 output.append(" DISC: " + str(i['DISC']))
73 output.append(" TRACK: " + str(i['TRACK']))
74 output.append(" DATE: " + str(i['DATE']))
75 output.append(" RESOLUTION: " + str(i['RESOLUTION']))
76 output.append(" THUMBNAIL: " + str(i['THUMBNAIL']))
77 output.append(" ALBUM_ART: " + str(i['ALBUM_ART']))
78 output.append(" ROTATION: " + str(i['ROTATION']))
79 output.append(" DLNA_PN: " + str(i['DLNA_PN']))
80 output.append(" MIME: " + str(i['MIME']))
81 output.append("")
82 textbox.insert(INSERT, "\n".join(output))
83
84bottom = Button(rightside, text="Submit", command=buttonCallback)
85rightside.add(bottom)
86mainloop()
87
server.py Raw
1# Requirements (rather, the versions of these requirements that I have):
2# Flask==0.10.1
3# Flask-Restless==0.12.0
4# Flask-SQLAlchemy==1.0
5# SQLAlchemy==0.8.2
6
7# Run this on the media server. It runs on port 5000 by default.
8
9# Copyright 2013 Steven Smith (blha303). All Rights Reserved.
10# New BSD license
11# http://www.opensource.org/licenses/BSD-3-Clause
12
13import flask
14import flask.ext.sqlalchemy
15app = flask.Flask(__name__)
16app.config['DEBUG'] = True
17app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:////var/lib/minidlna/files.db'
18#app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///data.db'
19db = flask.ext.sqlalchemy.SQLAlchemy(app)
20
21class Details(db.Model):
22 __tablename__ = "DETAILS"
23 ID = db.Column(db.Integer, primary_key=True)
24 PATH = db.Column(db.Text)
25 SIZE = db.Column(db.Integer)
26 TIMESTAMP = db.Column(db.Integer)
27 TITLE = db.Column(db.Text)
28 DURATION = db.Column(db.Text)
29 BITRATE = db.Column(db.Integer)
30 SAMPLERATE = db.Column(db.Integer)
31 CREATOR = db.Column(db.Text)
32 ARTIST = db.Column(db.Text)
33 ALBUM = db.Column(db.Text)
34 GENRE = db.Column(db.Text)
35 COMMENT = db.Column(db.Text)
36 CHANNELS = db.Column(db.Integer)
37 DISC = db.Column(db.Integer)
38 TRACK = db.Column(db.Integer)
39 DATE = db.Column(db.Date)
40 RESOLUTION = db.Column(db.Text)
41 THUMBNAIL = db.Column(db.Boolean)
42 ALBUM_ART = db.Column(db.Integer)
43 ROTATION = db.Column(db.Integer)
44 DLNA_PN = db.Column(db.Text)
45 MIME = db.Column(db.Text)
46
47import flask.ext.restless
48
49manager = flask.ext.restless.APIManager(app, flask_sqlalchemy_db=db)
50
51manager.create_api(Details, methods=['GET'], results_per_page=20)
52
53app.run(host='0.0.0.0')
54