diff options
Diffstat (limited to 'app/links')
-rw-r--r-- | app/links/__init__.py | 0 | ||||
-rw-r--r-- | app/links/admin.py | 14 | ||||
-rw-r--r-- | app/links/detail_urls.py | 13 | ||||
-rw-r--r-- | app/links/models.py | 74 | ||||
-rw-r--r-- | app/links/retriever.py | 111 | ||||
-rw-r--r-- | app/links/tumblr.py | 240 | ||||
-rw-r--r-- | app/links/urls.py | 15 |
7 files changed, 467 insertions, 0 deletions
diff --git a/app/links/__init__.py b/app/links/__init__.py new file mode 100644 index 0000000..e69de29 --- /dev/null +++ b/app/links/__init__.py diff --git a/app/links/admin.py b/app/links/admin.py new file mode 100644 index 0000000..2c73c76 --- /dev/null +++ b/app/links/admin.py @@ -0,0 +1,14 @@ +from django.contrib import admin + +from links.models import Link + +class LinkAdmin(admin.ModelAdmin): + list_display = ('title', 'rating', 'pub_date', 'status') + search_fields = ['title','description','url'] + list_filter = ['rating', 'status'] + fieldsets = ( + (None, {'fields': ('title','url','description','rating')}), + ('Details', {'fields': ('screen_url', 'pub_date', 'link_id', 'tags', 'status'), 'classes': 'collapse'}), + ) + +admin.site.register(Link, LinkAdmin)
\ No newline at end of file diff --git a/app/links/detail_urls.py b/app/links/detail_urls.py new file mode 100644 index 0000000..2b6ec5d --- /dev/null +++ b/app/links/detail_urls.py @@ -0,0 +1,13 @@ +from django.conf.urls.defaults import * +from django.views.generic.simple import redirect_to +from links.models import Link + + +detail_dict = { + 'queryset': Link.objects.filter(status__exact=1), +} +urlpatterns = patterns('django.views.generic.list_detail', + (r'^$', redirect_to, {'url': '/links/1/'}), + (r'^(?P<object_id>\d+)/$', 'object_detail', dict(detail_dict, template_name='details/link.html')), +) + diff --git a/app/links/models.py b/app/links/models.py new file mode 100644 index 0000000..4c3e59d --- /dev/null +++ b/app/links/models.py @@ -0,0 +1,74 @@ +import datetime +from django.db import models +from django.contrib.sitemaps import Sitemap +from django.contrib.syndication.views import Feed +from taggit.managers import TaggableManager + +RATINGS = ( + ('1', "1 Star"), + ('2', "2 Stars"), + ('3', "3 Stars"), + ('4', "4 Stars"), + ('5', "5 Stars"), +) + +DEBUG = 1 + +class Link(models.Model): + link_id = models.CharField(max_length=60, blank=True, null=True) + title = models.CharField(max_length=400) + url = models.CharField(max_length=400) + description = models.TextField(blank=True, null=True) + screen_url = models.CharField(max_length=400, blank=True, null=True) + rating = models.CharField(max_length=1, choices=RATINGS, null=True) + pub_date = models.DateTimeField() + PUB_STATUS = ( + (0, 'Private'), + (1, 'Public'), + ) + status = models.IntegerField(choices=PUB_STATUS, default=0) + tags = TaggableManager(blank=True) + + class Meta: + ordering = ['-pub_date'] + + def __unicode__(self): + return self.title + + def get_absolute_url(self): + return self.url + + def get_model_name(self): + return 'link' + + def get_previous_published(self): + return self.get_previous_by_pub_date(status__exact=1) + + def get_next_published(self): + return self.get_next_by_pub_date(status__exact=1) + + + def get_thumbnail_url(self): + return "http://images.luxagraf.net/magnolia_thumbs/%s" %(self.screen_url) + def comment_period_open(self): + return self.enable_comments and datetime.datetime.today() - datetime.timedelta(30) <= self.pub_date + + +class LinkSitemap(Sitemap): + changefreq = "never" + priority = 0.4 + + def items(self): + return Link.objects.filter(status=1) + + def lastmod(self, obj): + return obj.pub_date + +class LatestLinks(Feed): + title = "Luxagraf: Links" + link = "http://ma.gnolia.com/people/luxagraf/bookmarks" + description = "Links to interesting stuff" + description_template = 'feeds/links_description.html' + + def items(self): + return Link.objects.filter(status__exact=1).order_by('-pub_date')[:10] diff --git a/app/links/retriever.py b/app/links/retriever.py new file mode 100644 index 0000000..f876405 --- /dev/null +++ b/app/links/retriever.py @@ -0,0 +1,111 @@ +import datetime + +from django.core.exceptions import ObjectDoesNotExist +from django.template.defaultfilters import striptags +from django.core.mail import EmailMessage +from django.conf import settings + + +from utils.strutils import safestr,unquotehtml +from links.models import Link +from utils import pinboard + +def sync_pinboard_links(): + """ + sync bookmarks from my pinboard account + + dependancies: python-pinboard https://github.com/mgan59/python-pinboard/ + """ + p = pinboard.open(settings.PIN_USER, settings.PIN_PASS) + dupe = False + links = p.posts(count=30) + for link in links: + try: + #check to see if link exists + row = Link.objects.get(link_id=safestr(link['hash'])) + except ObjectDoesNotExist: + l, created = Link.objects.get_or_create( + title = link['description'], + link_id = safestr(link['hash']), + url = safestr(link['href']), + description = safestr(link['extended']), + rating = "3", + pub_date = datetime.datetime.strptime(link['time'], "%Y-%m-%dT%H:%M:%SZ"), + status = 0 + ) + print l.title + if created: + print l.title + for t in link['tags']: + l.tags.add(t) + email_link(l) + +''' +def sync_delicious_links(*args, **kwargs): + b = delicious.get_all(settings.DELICIOUS_USER, settings.DELICIOUS_PASS) + dupe = False + for post in b: + taglist = [] + try: + row = Link.objects.get(magnolia_id=safestr(post['hash'])) + # If the row exists already, set the dupe flag + dupe = True + except ObjectDoesNotExist: + #f = copy_file(safestr(post.findtext('screenshot')), safestr(info['id'])) + #fake the image since delicious doesn't offer them + local_image_url = "%s/%s.jpg" %(safestr(datetime.datetime.today().strftime("%b").lower()), safestr(post['hash'])) + tags = str(post['tag']).split(" ") + for tag in tags: + taglist.append(tag) + for tag in taglist: + if tag == '2lux': + status = 1 + break + else: + status = 0 + descr = markdown.markdown(unquotehtml(safestr(post['extended'])), safe_mode = False) + l, created = Link.objects.get_or_create( + title = post['description'], + magnolia_id = safestr(post['hash']), + url = safestr(post['href']), + description = descr, + screen_url = local_image_url, + #fake the rating since delicious doesn't offer such things + rating = "3", + pub_date = datetime.datetime.strptime(post['time'], "%Y-%m-%dT%H:%M:%SZ"), + status = status, + enable_comments = True, + tags = ", ".join(t for t in taglist if t != "2lux") + ) + + email_link(l) + if l.status == 1: + pass + #post_to_tumblr(l) + #send_to_deliciousfb(l) + if(dupe): + break + + +''' + +def email_link(link): + """ + Sends an imported link to Gmail (never hurts to have backups) + """ + subject = link.title + body = "%s\n\n%s\n\n\nvisit site:%s\n" %(link.title, link.description, link.url) + msg = EmailMessage(subject, striptags(body), 'sng@luxagraf.net', ['luxagraf+links@gmail.com']) + msg.send() + msg = EmailMessage(subject, striptags(body), 'sng@luxagraf.net', ['sng+links@luxagraf.net']) + msg.send() + +''' +def send_to_delicious(link): + del_tags = '' + tags = link.tags.split(',') + for tag in tags: + del_tags += tag.strip().replace(' ','_')+' ' + delicious.add(settings.DELICIOUS_USER, settings.DELICIOUS_PASS, link.url, link.title, tags = del_tags, extended = striptags(link.description), dt =safestr(link.pub_date), replace="no") + +'''
\ No newline at end of file diff --git a/app/links/tumblr.py b/app/links/tumblr.py new file mode 100644 index 0000000..8c0e286 --- /dev/null +++ b/app/links/tumblr.py @@ -0,0 +1,240 @@ +#!/usr/bin/python
+# -*- coding: utf-8 -*-
+
+# Copyright 2008 Ryan Cox ( ryan.a.cox@gmail.com ) All Rights Reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+# http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+#
+
+'''A wrapper library for Tumblr's public web API: http://www.tumblr.com/api'''
+
+__author__ = 'ryan.a.cox@gmail.com'
+__version__ = '0.1'
+
+
+from urllib2 import Request, urlopen, URLError, HTTPError
+from urllib import urlencode, quote
+import base64
+import re
+
+try:
+ import simplejson
+except ImportError:
+ from django.utils import simplejson
+
+GENERATOR = 'python-tumblr'
+PAGESIZE = 50
+
+
+class TumblrError(Exception):
+ ''' General Tumblr error '''
+ def __init__(self, msg):
+ self.msg = msg
+
+ def __str__(self):
+ return self.msg
+
+class TumblrAuthError(TumblrError):
+ ''' Wraps a 403 result '''
+ pass
+
+class TumblrRequestError(TumblrError):
+ ''' Wraps a 400 result '''
+ pass
+
+class TumblrIterator:
+ def __init__(self, name, start, max, type):
+ self.name = name
+ self.start = start
+ self.max = max
+ self.type = type
+ self.results = None
+ self.index = 0
+
+ def __iter__(self):
+ return self
+
+ def next(self):
+ if not self.results or (self.index == len(self.results['posts'])):
+ self.start += self.index
+ self.index = 0
+ url = "http://%s.tumblr.com/api/read/json?start=%s&num=%s" % (self.name,self.start, PAGESIZE)
+ if self.type:
+ url += "&type=" + self.type
+ response = urlopen(url)
+ page = response.read()
+ m = re.match("^.*?({.*}).*$", page,re.DOTALL | re.MULTILINE | re.UNICODE)
+ self.results = simplejson.loads(m.group(1))
+
+ if (self.index >= self.max) or len(self.results['posts']) == 0:
+ raise StopIteration
+
+ self.index += 1
+ return self.results['posts'][self.index-1]
+
+class Api:
+ def __init__(self, name, email=None, password=None ):
+ self.name = name
+ self.is_authenticated = False
+ self.email = email
+ self.password = password
+
+ def auth_check(self):
+ if self.is_authenticated:
+ return
+ url = 'http://www.tumblr.com/api/write'
+ values = {
+ 'action': 'authenticate',
+ 'generator' : GENERATOR,
+ 'email': self.email,
+ 'password' : self.password }
+
+ data = urlencode(values)
+ req = Request(url, data)
+ try:
+ response = urlopen(req)
+ page = response.read()
+ self.url = page
+ self.is_authenticated = True
+ return
+ except HTTPError, e:
+ if 403 == e.code:
+ raise TumblrAuthError(str(e))
+ if 400 == e.code:
+ raise TumblrRequestError(str(e))
+ except Exception, e:
+ raise TumblrError(str(e))
+
+
+ def write_regular(self, title=None, body=None, **args):
+ if title:
+ args['title'] = title
+ if body:
+ args['body'] = body
+ args = self._fixnames(args)
+ if not 'title' in args and not 'body' in args:
+ raise TumblrError("Must supply either body or title argument")
+
+ self.auth_check()
+ args['type'] = 'regular'
+ return self._write(args)
+
+ def write_photo(self, source=None, **args):
+ if source:
+ args['source'] = source
+
+ args = self._fixnames(args)
+ if 'source' in args and 'data' in args:
+ raise TumblrError("Must NOT supply both source and data arguments")
+
+ if not 'source' in args and not 'data' in args:
+ raise TumblrError("Must supply source or data argument")
+
+ self.auth_check()
+ args['type'] = 'photo'
+ return self._write(args)
+
+ def write_quote(self, quote=None, **args):
+ if quote:
+ args['quote'] = quote
+ args = self._fixnames(args)
+ if not 'quote' in args:
+ raise TumblrError("Must supply quote arguments")
+
+ self.auth_check()
+ args['type'] = 'quote'
+ return self._write(args)
+
+ def write_link(self, url=None, **args):
+ if url:
+ args['url'] = url
+ args = self._fixnames(args)
+ if not 'url' in args:
+ raise TumblrError("Must supply url argument")
+
+ self.auth_check()
+ args['type'] = 'link'
+ return self._write(args)
+
+ def write_conversation(self, conversation=None, **args):
+ if conversation:
+ args['conversation'] = conversation
+ args = self._fixnames(args)
+ if not 'conversation' in args:
+ raise TumblrError("Must supply conversation argument")
+
+ self.auth_check()
+ args['type'] = 'conversation'
+ return self._write(args)
+
+ def write_video(self, embed=None, **args):
+ if embed:
+ args['embed'] = embed
+ args = self._fixnames(args)
+ if 'embed' in args and 'data' in args:
+ raise TumblrError("Must NOT supply both embed and data arguments")
+
+ if not 'embed' in args and not 'data' in args:
+ raise TumblrError("Must supply embed or data argument")
+
+ self.auth_check()
+ args['type'] = 'video'
+ return self._write(args)
+
+ def _fixnames(self, args):
+ for key in args:
+ if '_' in key:
+ value = args[key]
+ del args[key]
+ args[key.replace('_', '-')] = value
+ return args
+
+ def _write(self, params, headers=None):
+ self.auth_check()
+ url = 'http://www.tumblr.com/api/write'
+ params['email'] = self.email
+ params['password'] = self.password
+ params['generator'] = GENERATOR
+ data = urlencode(params)
+ if headers:
+ req = Request(url, data, headers)
+ else:
+ req = Request(url, data)
+ newid = None
+ try:
+ urlopen(req)
+ raise TumblrError("Error writing post")
+
+ except HTTPError, e:
+ if 201 == e.code:
+ newid = e.read()
+ return self.read(id=newid)
+ raise TumblrError(e.read())
+
+ def read(self, id=None, start=0,max=2**31-1,type=None):
+ if id:
+ url = "http://%s.tumblr.com/api/read/json?id=%s" % (self.name,id)
+ print url
+ response = urlopen(url)
+ page = response.read()
+ m = re.match("^.*?({.*}).*$", page,re.DOTALL | re.MULTILINE | re.UNICODE)
+ results = simplejson.loads(m.group(1))
+ if len(results['posts']) == 0:
+ return None
+
+ return results['posts'][0]
+ else:
+ return TumblrIterator(self.name,start,max,type)
+
+if __name__ == "__main__":
+ pass
diff --git a/app/links/urls.py b/app/links/urls.py new file mode 100644 index 0000000..5a16d6e --- /dev/null +++ b/app/links/urls.py @@ -0,0 +1,15 @@ +from django.conf.urls.defaults import * +from django.views.generic import list_detail +from django.views.generic.simple import redirect_to +from links.models import Link + +photos_paged = { + 'queryset': Link.objects.filter(status__exact=1).order_by('-pub_date'), + 'paginate': True, + 'page_url': '/links/%d/', +} + +urlpatterns = patterns('', + (r'^(?P<page>\d+)/$', list_detail.object_list, dict(photos_paged, template_name='archives/links.html')), + (r'^$', redirect_to, {'url': '/links/1/'}), +) |